Last active
May 8, 2023 01:52
-
-
Save MattiasMartens/4bc8c9ca515a566fa1afd05355bfb239 to your computer and use it in GitHub Desktop.
Bluesky Threader
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!DOCTYPE html> | |
<html> | |
<head> | |
<title>Tweet Threader</title> | |
<link | |
href="" | |
rel="icon" | |
type="image/x-icon" | |
/> | |
<style> | |
.col1, | |
.col2 { | |
display: inline-block; | |
width: 30em; | |
vertical-align: top; | |
} | |
.row { | |
margin: auto; | |
width: fit-content; | |
} | |
textarea { | |
width: 28em; | |
box-sizing: border-box; | |
padding: 0.5em; | |
margin-left: 1em; | |
} | |
#char-count { | |
display: inline-block; | |
float: right; | |
margin-right: 1em; | |
} | |
/** circle */ | |
#char-count .quiet-remaining:before { | |
content: "•"; | |
font-size: large; | |
color: #1da1f2; | |
cursor: pointer; | |
} | |
#char-count .quiet-remaining.near-edge:before { | |
color: red; | |
} | |
.tweet-block--content, | |
textarea, | |
.tweet-block--index, | |
.toast, | |
#char-count, | |
p { | |
font-family: "Lucida Console", Monaco, monospace; | |
font-size: 1em; | |
} | |
h1 { | |
font-family: "system-ui", -apple-system, BlinkMacSystemFont, "Segoe UI", | |
Roboto, "Helvetica Neue", Arial, sans-serif; | |
color: #1da1f2; | |
text-align: center; | |
margin-bottom: 0; | |
} | |
.subtitle { | |
text-align: center; | |
font-family: "system-ui", -apple-system, BlinkMacSystemFont, "Segoe UI", | |
Roboto, "Helvetica Neue", Arial, sans-serif; | |
} | |
h2 { | |
font-family: "system-ui", -apple-system, BlinkMacSystemFont, "Segoe UI", | |
Roboto, "Helvetica Neue", Arial, sans-serif; | |
color: #125580; | |
} | |
.tweet-block { | |
border: 1px solid #ccc; | |
position: relative; | |
margin-bottom: 1em; | |
background-color: #f7fcff; | |
width: 30em; | |
} | |
.tweet-block:hover { | |
background-color: #e8f5fe; | |
} | |
.tweet-block--index { | |
position: absolute; | |
top: 10px; | |
left: 10px; | |
color: #1da1f2; | |
} | |
.tweet-block--content { | |
width: 26em; | |
margin-left: 2em; | |
margin-top: 2.5em; | |
margin-bottom: 1.5em; | |
padding: 0.25em; | |
display: inline-block; | |
cursor: pointer; | |
cursor: copy; | |
transition: background-color 250ms; | |
word-wrap: break-word; | |
} | |
.tweet-block--content p { | |
margin-top: 1em; | |
} | |
.tweet-block--content:hover { | |
background-color: #9fd2f2; | |
} | |
.toast { | |
background-color: #1da1f2; | |
color: white; | |
padding: 1em; | |
position: fixed; | |
bottom: 1em; | |
right: 1em; | |
z-index: 1; | |
opacity: 0; | |
transition: opacity 250ms; | |
} | |
code { | |
background-color: #f7fcff; | |
padding: 0.25em; | |
margin: 0; | |
border-radius: 0.25em; | |
/** red code color */ | |
color: #e83e8c; | |
/** light grey border */ | |
border: 1px solid #d1d5da; | |
margin-left: -0.25em; | |
margin-right: -0.25em; | |
} | |
</style> | |
</head> | |
<body> | |
<h1>Tweet Threader</h1> | |
<div class="subtitle"> | |
By | |
<a href="https://twitter.com/mattiasinspace" _target="blank" | |
>@MattiasInSpace</a | |
> | |
</div> | |
<div class="row"> | |
<div class="col1"> | |
<h2>Enter text you want to make into a thread</h2> | |
<p> | |
Text will break into tweets every 280 characters. To add an explicit | |
tweet break, add the pipe character <code>|</code> on its own line. | |
</p> | |
<textarea id="input-text"></textarea> | |
<span id="char-count"></span> | |
</div> | |
<div class="col2"> | |
<div id="output"> | |
<h2>Rendered into tweet blocks</h2> | |
<div id="output-text"></div> | |
</div> | |
</div> | |
</div> | |
<script> | |
window.toggleAlwaysShowRemaining = function () { | |
const newAlwaysShowRemaining = !localStorage.getItem( | |
"alwaysShowRemaining" | |
); | |
const remainingElement = document.querySelector(".remaining"); | |
if (remainingElement) { | |
if (newAlwaysShowRemaining) { | |
remainingElement.style.display = "initial"; | |
} else if (!remainingElement.classList.contains("always-show")) { | |
remainingElement.style.display = "none"; | |
} | |
} | |
newAlwaysShowRemaining ? localStorage.setItem("alwaysShowRemaining", 'true') : localStorage.removeItem("alwaysShowRemaining"); | |
}; | |
function truncateTextWithEllipsis(text, maxLength) { | |
if (text.length <= maxLength) { | |
return text; | |
} | |
return text.slice(0, maxLength - 1) + "…"; | |
} | |
function showToast(message) { | |
// Create the toast element | |
const toast = document.createElement("div"); | |
toast.classList.add("toast"); | |
toast.innerHTML = message; | |
// Insert the toast element into the DOM | |
document.body.appendChild(toast); | |
// Show the toast | |
setTimeout(function () { | |
toast.style.opacity = 1; | |
}, 0); | |
// Show the toast for 3 seconds | |
setTimeout(function () { | |
toast.style.opacity = 0; | |
setTimeout(function () { | |
toast.remove(); | |
}, 250); | |
}, 3000); | |
} | |
function escapeHTML(html) { | |
return html.replace(/[&<>"]/g, function (char) { | |
switch (char) { | |
case "&": | |
return "&"; | |
case "<": | |
return "<"; | |
case ">": | |
return ">"; | |
case '"': | |
return """; | |
} | |
}); | |
} | |
const rerenderInput = function () { | |
// Break input text into Tweets | |
/** @type string */ const text = inputText.value; | |
const delimited = text.split("\n|\n"); | |
// Use newlines as paragraph breaks | |
const brokenLengthwise = delimited | |
.map((chunk) => { | |
const broken = chunk.match(/.{1,280}/gs); | |
if (!broken) { | |
return []; | |
} | |
return [...broken].map((chunk) => | |
chunk | |
.split("\n") | |
.map((innerChunk) => `${escapeHTML(innerChunk)}<br>`) | |
.join("") | |
); | |
}) | |
.flat(); | |
const outputHTML = outputText.innerHTML; | |
const marker = '<span style="color: red;">|</span>'; | |
outputText.innerHTML = brokenLengthwise | |
.map( | |
(tweet, index, { length }) => | |
`<div class="tweet-block"><div class="tweet-block--index">${ | |
index + 1 | |
}/${length}</div><div class="tweet-block--content" title="Click to copy">${tweet}</div></div>` | |
) | |
.join(""); | |
// Save previous input to local storage | |
localStorage.setItem("inputText", inputText.value); | |
// Update height of textarea input | |
inputText.style.height = "1px"; | |
inputText.style.height = 25 + inputText.scrollHeight + "px"; | |
// Update char count | |
const charCount = document.getElementById("char-count"); | |
charCount.innerHTML = ""; | |
const lastTweet = document.querySelector( | |
".tweet-block:last-child .tweet-block--content" | |
); | |
if (lastTweet) { | |
const innerTextLength = lastTweet.innerText.length - 1; | |
const remaining = 280 - innerTextLength; | |
const showRemaining = remaining <= 20; | |
const showRemainingMerged = showRemaining || !!localStorage.getItem("alwaysShowRemaining"); | |
const nearEdge = remaining <= 10; | |
const charText = `${innerTextLength}/280 (remaining: ${remaining})`; | |
const quietRemainingHtml = `<span class="quiet-remaining ${ | |
nearEdge ? "near-edge" : "" | |
}" title="${charText}" onclick="window.toggleAlwaysShowRemaining()"></span>`; | |
const color = nearEdge ? "red" : "#125580"; | |
const remainingHtml = `<span class="remaining ${showRemaining ? 'always-show' : ''}" style="color: ${color}; margin-left: 0.5em; display: ${ | |
showRemainingMerged ? "initial" : "none" | |
}">${charText}</span>`; | |
charCount.innerHTML = quietRemainingHtml + remainingHtml; | |
} | |
}; | |
const inputText = document.getElementById("input-text"); | |
const outputText = document.getElementById("output-text"); | |
inputText.addEventListener("input", rerenderInput); | |
// Load previous input from local storage | |
const previousInput = localStorage.getItem("inputText"); | |
if (previousInput) { | |
inputText.value = previousInput; | |
rerenderInput(); | |
} | |
// Add click-to-copy for tweet blocks | |
const col2 = document.querySelector(".col2"); | |
col2.addEventListener("click", (event) => { | |
const target = event.target; | |
const content = target.closest(".tweet-block--content"); | |
if (!!target) { | |
const range = document.createRange(); | |
range.selectNode(target); | |
window.getSelection().removeAllRanges(); | |
window.getSelection().addRange(range); | |
document.execCommand("copy"); | |
window.getSelection().removeAllRanges(); | |
showToast( | |
`Copied to clipboard: ${truncateTextWithEllipsis( | |
content.innerText, | |
32 | |
)}` | |
); | |
} | |
}); | |
</script> | |
</body> | |
</html> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment