Skip to content

Instantly share code, notes, and snippets.

@MattiasMartens
Last active May 8, 2023 01:52
Show Gist options
  • Save MattiasMartens/4bc8c9ca515a566fa1afd05355bfb239 to your computer and use it in GitHub Desktop.
Save MattiasMartens/4bc8c9ca515a566fa1afd05355bfb239 to your computer and use it in GitHub Desktop.
Bluesky Threader
<!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 "&amp;";
case "<":
return "&lt;";
case ">":
return "&gt;";
case '"':
return "&quot;";
}
});
}
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