js/mod/recent_posts.js: add recent_posts.js

This commit is contained in:
Jon 2024-01-30 18:47:20 +00:00 committed by towards-a-new-leftypol
parent 51d6374d85
commit 960919c2dd
1 changed files with 117 additions and 0 deletions

117
js/mod/recent_posts.js Normal file
View File

@ -0,0 +1,117 @@
/**
* Implements live-reloading on the recent posts mod page.
*/
(() => {
const kLocationPath = location.href.slice(location.origin.length)
if (/^\/mod\.php\?\/recent\/\d+$/.test(kLocationPath)) {
const loadIntFromStorage = key => {
const value = localStorage.getItem(`jon-liveposts-enabled::${key}`)
return value != null ? parseInt(value) : null
}
const kPullXPosts = loadIntFromStorage("pull-x-posts") ?? 6
const kPullEveryX = loadIntFromStorage("pull-every-x") ?? 8000
const kRecentXPosts = parseInt(kLocationPath.slice(17).split(/[^\d]/)[0])
let liveUpdateEnabled = false
let liveUpdateAbortController = null
let liveUpdateIntervalId = null
let liveUpdateFaviconChanged = false
const parser = new DOMParser()
const fetchLatest = async () => {
const res = await fetch(`/mod.php?/recent/${kPullXPosts}`, {
"signal": AbortSignal.any([ liveUpdateAbortController.signal, AbortSignal.timeout(kPullEveryX - 500) ])
})
if (res.ok) {
const dom = parser.parseFromString(await res.text(), "text/html")
const posts = Array.from(dom.querySelectorAll("body > .post-wrapper[data-board]"))
return posts.map(el => ({
"element": el,
"href": Array.prototype.find.apply(el.children, [ el => el.classList.contains("eita-link") ]).href,
"id": Array.prototype.find.apply(el.children, [ el => el.classList.contains("thread") || el.classList.contains("postcontainer") ]).id
}))
} else {
return []
}
}
const getPostTimestamp = post => new Date(post.lastElementChild.querySelector(".intro time").dateTime)
const updateRecentPosts = async () => {
if (liveUpdateEnabled) {
const lastPost = document.body.querySelector(".post-wrapper")
const lastPostTs = getPostTimestamp(lastPost).getTime()
const posts = (await fetchLatest()).filter(post => document.getElementById(post.id) == null && getPostTimestamp(post.element).getTime() > lastPostTs)
if (posts.length > 0) {
const totalPosts = Array.from(document.body.querySelectorAll(".post-wrapper"))
for (const post of posts) {
lastPost.prepend(post.element)
totalPosts.unshift(post.element)
}
while (totalPosts.length > kRecentXPosts) {
document.body.removeChild(totalPosts.pop())
}
updatePageNext()
makeIcon("reply")
liveUpdateFaviconChanged = true
}
}
}
const pageNext = Array.prototype.find.apply(document.body.children, [ el => el.nodeName == "A" && el.href.startsWith(`${location.origin}/mod.php?/recent/${kRecentXPosts}&last=`) ])
const updatePageNext = () => {
const posts = document.body.querySelectorAll(".post-wrapper")
const oldestPost = posts[posts.length-1]
pageNext.href = `/mod.php?/recent/${kRecentXPosts}&last=${getPostTimestamp(oldestPost).getTime() / 1000}`
}
const createCheckbox = () => {
const id = "jon-liveposts-enabled"
const div = document.createElement("div")
const label = document.createElement("label")
const checkbox = document.createElement("input")
checkbox.type = "checkbox"
checkbox.id = id
label.innerText = "live update: "
label.for = id
div.style.display = "inline"
div.append(" ")
div.appendChild(label)
div.appendChild(checkbox)
div.append(" ")
return { checkbox, label, div }
}
const pageNums = Array.prototype.find.apply(document.body.children, [ el => el.nodeName == "P" ])
const form = createCheckbox()
form.checkbox.addEventListener("click", () => {
setTimeout(() => {
if (form.checkbox.checked) {
liveUpdateEnabled = true
liveUpdateAbortController = new AbortController()
liveUpdateIntervalId = setInterval(() => updateRecentPosts(), kPullEveryX)
} else {
liveUpdateEnabled = false
clearInterval(liveUpdateIntervalId)
liveUpdateAbortController.abort()
liveUpdateIntervalId = null
liveUpdateAbortController = null
}
}, 700)
})
document.body.addEventListener("mousemove", () => {
if (liveUpdateFaviconChanged) {
liveUpdateFaviconChanged = false
makeIcon(false)
}
})
pageNums.append("|")
pageNums.append(form.div)
}
})()