atlas/web/_js/main.js

554 lines
13 KiB
JavaScript
Raw Normal View History

/*
========================================================================
2022-04-28 12:41:19 +02:00
The 2022 r/place Atlas
An atlas of Reddit's 2022 r/place, with information to each
artwork of the canvas provided by the community.
Copyright (c) 2017 Roland Rytz <roland@draemm.li>
Copyright (c) 2022 Place Atlas contributors
Licensed under the GNU Affero General Public License Version 3
2022-04-09 03:23:15 +02:00
https://place-atlas.stefanocoding.me/license.txt
========================================================================
*/
2022-04-09 03:23:15 +02:00
const prodDomain = "place-atlas.stefanocoding.me"
2022-05-06 09:41:22 +02:00
const innerContainer = document.getElementById("innerContainer")
const container = document.getElementById("container")
const canvas = document.getElementById("highlightCanvas")
const context = canvas.getContext("2d")
2022-05-06 09:41:22 +02:00
let zoom = 1
if (window.devicePixelRatio) {
2022-05-06 09:41:22 +02:00
zoom = 1 / window.devicePixelRatio
}
2022-05-06 09:41:22 +02:00
const maxZoom = 128
const minZoom = 0.1
2022-05-06 09:41:22 +02:00
let zoomOrigin = [0, 0]
let scaleZoomOrigin = [0, 0]
2022-05-06 09:41:22 +02:00
let dragging = false
let lastPosition = [0, 0]
2022-05-06 09:41:22 +02:00
const viewportSize = [0, 0]
function applyView() {
2022-05-06 09:41:22 +02:00
//console.log(zoomOrigin, scaleZoomOrigin)
//console.log(scaleZoomOrigin[0])
2017-04-10 16:28:22 +02:00
2022-05-06 09:41:22 +02:00
scaleZoomOrigin[0] = Math.max(-1000, Math.min(1000, scaleZoomOrigin[0]))
scaleZoomOrigin[1] = Math.max(-1000, Math.min(1000, scaleZoomOrigin[1]))
2017-04-10 16:28:22 +02:00
2022-05-06 09:41:22 +02:00
zoomOrigin = [scaleZoomOrigin[0] * zoom, scaleZoomOrigin[1] * zoom]
2022-05-06 09:41:22 +02:00
innerContainer.style.height = (~~(zoom * 2000)) + "px"
innerContainer.style.width = (~~(zoom * 2000)) + "px"
2022-05-06 09:41:22 +02:00
innerContainer.style.left = ~~(container.clientWidth / 2 - innerContainer.clientWidth / 2 + zoomOrigin[0] + container.offsetLeft) + "px"
innerContainer.style.top = ~~(container.clientHeight / 2 - innerContainer.clientHeight / 2 + zoomOrigin[1] + container.offsetTop) + "px"
}
2022-05-06 09:41:22 +02:00
let atlas = null
2022-04-10 11:11:34 +02:00
window.atlas = atlas
let atlasAll = null
2022-04-10 11:11:34 +02:00
window.atlasAll = atlasAll
2022-04-05 11:48:15 +02:00
2022-04-08 17:35:21 +02:00
if (document.location.host !== prodDomain) document.body.dataset.dev = ""
2022-05-06 09:41:22 +02:00
init()
async function init() {
2022-04-06 17:34:04 +02:00
// For Reviewing Reddit Changes
2022-05-06 09:41:22 +02:00
//let resp = await fetch("../tools/temp_atlas.json")
const resp = await fetch("./atlas.json")
atlas = await resp.json()
2022-04-05 11:48:15 +02:00
atlas.sort(function (a, b) {
if (a.center[1] < b.center[1]) {
2022-05-06 09:41:22 +02:00
return -1
2022-04-05 11:48:15 +02:00
}
if (a.center[1] > b.center[1]) {
2022-05-06 09:41:22 +02:00
return 1
2022-04-05 11:48:15 +02:00
}
// a must be equal to b
2022-05-06 09:41:22 +02:00
return 0
})
2022-05-06 09:41:22 +02:00
atlasAll = updateAtlasAll(atlas)
2022-05-06 09:41:22 +02:00
let mode = "view"
2022-05-06 09:41:22 +02:00
const args = window.location.search
const params = new URLSearchParams(args)
if (args) {
mode = params.get("mode")
if (!mode) {
2022-05-06 09:41:22 +02:00
mode = "view"
}
2022-04-09 21:09:57 +02:00
// Backwards compatibility for old links using "search" id arg
if (params.has('id') && params.get('mode') !== 'draw') {
2022-04-16 14:26:31 +02:00
const id = params.get('id')
params.delete('id')
2022-04-16 14:26:31 +02:00
const newLocation = new URL(window.location)
newLocation.hash = id
newLocation.search = params
window.history.replaceState({}, '', newLocation)
2022-04-09 21:09:57 +02:00
}
}
const hash = window.location.hash.substring(1)
2022-05-06 09:41:22 +02:00
const [, period] = hash.split('/')
if (period) {
2022-05-06 09:41:22 +02:00
const [, targetPeriod, targetVariation] = parsePeriod(period)
await updateTime(targetPeriod, targetVariation)
} else {
await updateTime(currentPeriod, currentVariation)
}
2022-05-06 09:41:22 +02:00
//console.log(document.documentElement.clientWidth, document.documentElement.clientHeight)
2022-05-06 09:41:22 +02:00
zoomOrigin = [0, 0]
applyView()
2022-05-06 09:41:22 +02:00
let initialPinchDistance = 0
let initialPinchZoom = 0
let initialPinchZoomOrigin = [0, 0]
2022-05-06 09:41:22 +02:00
let desiredZoom
let zoomAnimationFrame
document.body.dataset.mode = mode
2022-04-09 11:27:14 +02:00
initGlobal()
if (mode !== "draw") initViewGlobal()
if (mode === "draw") {
2022-05-06 09:41:22 +02:00
initDraw()
} else if (mode === "about") {
2022-05-06 09:41:22 +02:00
window.location = "./about.html"
} else if (mode === "overlap") {
if (initOverlap) {
2022-05-06 09:41:22 +02:00
initOverlap()
}
} else if (mode.startsWith("diff")) {
2022-04-08 06:06:16 +02:00
try {
2022-05-06 09:41:22 +02:00
const liveResp = await fetch("https://place-atlas.stefanocoding.me/atlas.json")
let liveJson = await liveResp.json()
liveJson = updateAtlasAll(liveJson)
const liveAtlasReduced = liveJson.reduce(function (a, c) {
2022-05-06 09:41:22 +02:00
a[c.id] = c
return a
}, {})
2022-04-08 22:02:14 +02:00
// Mark added/edited entries
atlasAll = atlasAll.map(function (entry) {
if (liveAtlasReduced[entry.id] === undefined) {
2022-05-06 09:41:22 +02:00
entry.diff = "add"
} else if (JSON.stringify(entry) !== JSON.stringify(liveAtlasReduced[entry.id])) {
2022-05-06 09:41:22 +02:00
entry.diff = "edit"
2022-04-08 06:06:16 +02:00
}
2022-05-06 09:41:22 +02:00
return entry
})
2022-04-08 22:02:14 +02:00
// Mark removed entries
const atlasReduced = atlasAll.reduce(function (a, c) {
2022-05-06 09:41:22 +02:00
a[c.id] = c
return a
}, {})
const removedEntries = liveJson.filter(entry =>
atlasReduced[entry.id] === undefined
).map(entry => {
entry.diff = "delete"
return entry
})
atlasAll.push(...removedEntries)
2022-04-08 22:02:14 +02:00
if (mode.includes("only")) {
atlasAll = atlasAll.filter(function (entry) {
2022-04-08 22:02:14 +02:00
return typeof entry.diff == "string"
2022-05-06 09:41:22 +02:00
})
2022-04-08 22:02:14 +02:00
}
2022-04-08 06:06:16 +02:00
} catch (error) {
2022-05-06 09:41:22 +02:00
console.warn("Diff mode failed to load, reverting to normal view.", error)
2022-04-08 06:06:16 +02:00
} finally {
2022-04-16 14:51:51 +02:00
await updateTime()
if (initOverlap && mode.includes("overlap")) {
2022-05-06 09:41:22 +02:00
initOverlap()
2022-04-08 22:02:14 +02:00
} else {
2022-05-06 09:41:22 +02:00
initView()
2022-04-08 22:02:14 +02:00
}
2022-04-08 06:06:16 +02:00
}
} else if (mode === "explore") {
2022-05-06 09:41:22 +02:00
initExplore()
} else {
2022-05-06 09:41:22 +02:00
initView()
}
2017-04-10 17:57:13 +02:00
2022-05-06 09:41:22 +02:00
document.getElementById("loading").classList.add("d-none")
2017-04-10 17:57:13 +02:00
document.getElementById("zoomInButton").addEventListener("click", function (e) {
2017-04-10 12:25:04 +02:00
/*if(zoomAnimationFrame){
2022-05-06 09:41:22 +02:00
window.cancelAnimationFrame(zoomAnimationFrame)
2017-04-10 12:25:04 +02:00
}*/
2022-05-06 09:41:22 +02:00
const x = container.clientWidth / 2
const y = container.clientHeight / 2
initialPinchZoomOrigin = [
scaleZoomOrigin[0],
scaleZoomOrigin[1]
2022-05-06 09:41:22 +02:00
]
2022-05-06 09:41:22 +02:00
initialPinchZoom = zoom
2022-05-06 09:41:22 +02:00
lastPosition = [x, y]
zoom = zoom * 2
zoom = Math.max(minZoom, Math.min(maxZoom, zoom))
2022-05-06 09:41:22 +02:00
applyZoom(x, y, zoom)
2022-05-06 09:41:22 +02:00
})
2017-04-05 00:53:10 +02:00
document.getElementById("zoomOutButton").addEventListener("click", function (e) {
2017-04-10 12:25:04 +02:00
/*if(zoomAnimationFrame){
2022-05-06 09:41:22 +02:00
window.cancelAnimationFrame(zoomAnimationFrame)
2017-04-10 12:25:04 +02:00
}*/
2022-05-06 09:41:22 +02:00
const x = container.clientWidth / 2
const y = container.clientHeight / 2
initialPinchZoomOrigin = [
scaleZoomOrigin[0],
scaleZoomOrigin[1]
2022-05-06 09:41:22 +02:00
]
2022-05-06 09:41:22 +02:00
initialPinchZoom = zoom
2022-05-06 09:41:22 +02:00
lastPosition = [x, y]
zoom = zoom / 2
zoom = Math.max(minZoom, Math.min(maxZoom, zoom))
2022-05-06 09:41:22 +02:00
applyZoom(x, y, zoom)
})
2017-04-05 00:53:10 +02:00
document.getElementById("zoomResetButton").addEventListener("click", function (e) {
2022-05-06 09:41:22 +02:00
zoom = 1
zoomOrigin = [0, 0]
scaleZoomOrigin = [0, 0]
updateLines()
applyView()
})
2017-04-05 00:53:10 +02:00
container.addEventListener("dblclick", function (e) {
2017-04-10 12:25:04 +02:00
/*if(zoomAnimationFrame){
2022-05-06 09:41:22 +02:00
window.cancelAnimationFrame(zoomAnimationFrame)
2017-04-10 12:25:04 +02:00
}*/
2022-05-06 09:41:22 +02:00
const x = e.clientX - container.offsetLeft
const y = e.clientY - container.offsetTop
initialPinchZoomOrigin = [
scaleZoomOrigin[0],
scaleZoomOrigin[1]
2022-05-06 09:41:22 +02:00
]
2022-05-06 09:41:22 +02:00
initialPinchZoom = zoom
2022-05-06 09:41:22 +02:00
lastPosition = [x, y]
if (e.ctrlKey) {
2022-05-06 09:41:22 +02:00
zoom = zoom / 2
} else {
2022-05-06 09:41:22 +02:00
zoom = zoom * 2
}
2022-05-06 09:41:22 +02:00
zoom = Math.max(minZoom, Math.min(maxZoom, zoom))
applyZoom(x, y, zoom)
2022-05-06 09:41:22 +02:00
e.preventDefault()
})
container.addEventListener("wheel", function (e) {
2017-04-10 12:25:04 +02:00
/*if(zoomAnimationFrame){
2022-05-06 09:41:22 +02:00
window.cancelAnimationFrame(zoomAnimationFrame)
2017-04-10 12:25:04 +02:00
}*/
2022-05-06 09:41:22 +02:00
const x = e.clientX - container.offsetLeft
const y = e.clientY - container.offsetTop
initialPinchZoomOrigin = [
scaleZoomOrigin[0],
scaleZoomOrigin[1]
2022-05-06 09:41:22 +02:00
]
2022-05-06 09:41:22 +02:00
initialPinchZoom = zoom
2022-05-06 09:41:22 +02:00
lastPosition = [x, y]
2022-04-06 22:42:48 +02:00
// Check if we are zooming by pixels
// https://developer.mozilla.org/en-US/docs/Web/API/WheelEvent/deltaMode
if (e.deltaMode === 0) {
// Scale the pixel delta by the current zoom factor
// We want to zoom faster when closer, and slower when further
// This creates a smoother experience
2022-05-06 09:41:22 +02:00
zoom -= e.deltaY * (0.001 * zoom)
2022-04-06 22:42:48 +02:00
} else {
if (e.deltaY > 0) {
2022-05-06 09:41:22 +02:00
zoom = zoom / 2
} else if (e.deltaY < 0) {
2022-05-06 09:41:22 +02:00
zoom = zoom * 2
2022-04-06 22:42:48 +02:00
}
}
2022-05-06 09:41:22 +02:00
zoom = Math.max(minZoom, Math.min(maxZoom, zoom))
applyZoom(x, y, zoom)
}, { passive: true })
2017-04-10 12:25:04 +02:00
/*function setDesiredZoom(x, y, target){
2022-05-06 09:41:22 +02:00
zoom = (zoom*2 + target)/3
//console.log(zoom)
if(Math.abs(1 - zoom/target) <= 0.01){
2022-05-06 09:41:22 +02:00
zoom = target
}
2022-05-06 09:41:22 +02:00
applyZoom(x, y, zoom)
if(zoom != target){
zoomAnimationFrame = window.requestAnimationFrame(function(){
2022-05-06 09:41:22 +02:00
setDesiredZoom(x, y, target)
})
}
2017-04-10 12:25:04 +02:00
}*/
container.addEventListener("mousedown", function (e) {
2022-05-06 09:41:22 +02:00
mousedown(e.clientX, e.clientY)
e.preventDefault()
})
container.addEventListener("touchstart", function (e) {
if (e.touches.length == 2) {
2022-05-06 09:41:22 +02:00
e.preventDefault()
}
2022-05-06 09:41:22 +02:00
touchstart(e)
2022-05-06 09:41:22 +02:00
}, { passive: true })
function mousedown(x, y) {
2022-05-06 09:41:22 +02:00
lastPosition = [x, y]
dragging = true
}
function touchstart(e) {
if (e.touches.length == 1) {
2022-05-06 09:41:22 +02:00
mousedown(e.touches[0].clientX, e.touches[0].clientY)
} else if (e.touches.length == 2) {
initialPinchDistance = Math.sqrt(
Math.pow(e.touches[0].clientX - e.touches[1].clientX, 2)
+ Math.pow(e.touches[0].clientY - e.touches[1].clientY, 2)
2022-05-06 09:41:22 +02:00
)
2022-05-06 09:41:22 +02:00
initialPinchZoom = zoom
initialPinchZoomOrigin = [
scaleZoomOrigin[0],
scaleZoomOrigin[1]
2022-05-06 09:41:22 +02:00
]
mousedown(
(e.touches[0].clientX + e.touches[1].clientX) / 2,
(e.touches[0].clientY + e.touches[1].clientY) / 2
2022-05-06 09:41:22 +02:00
)
}
}
window.addEventListener("mousemove", function (e) {
2022-05-06 09:41:22 +02:00
updateLines()
mousemove(e.clientX, e.clientY)
if (dragging) {
2022-05-06 09:41:22 +02:00
e.preventDefault()
}
2022-05-06 09:41:22 +02:00
})
window.addEventListener("touchmove", function (e) {
if (e.touches.length == 2 || e.scale > 1) {
2022-05-06 09:41:22 +02:00
e.preventDefault()
}
2022-05-06 09:41:22 +02:00
touchmove(e)
},
{ passive: false }
2022-05-06 09:41:22 +02:00
)
function mousemove(x, y) {
if (dragging) {
container.style.cursor = "move"
2022-05-06 09:41:22 +02:00
const deltaX = x - lastPosition[0]
const deltaY = y - lastPosition[1]
lastPosition = [x, y]
2022-05-06 09:41:22 +02:00
zoomOrigin[0] += deltaX
zoomOrigin[1] += deltaY
2022-05-06 09:41:22 +02:00
scaleZoomOrigin[0] += deltaX / zoom
scaleZoomOrigin[1] += deltaY / zoom
2022-05-06 09:41:22 +02:00
previousZoomOrigin = [zoomOrigin[0], zoomOrigin[1]]
previousScaleZoomOrigin = [scaleZoomOrigin[0], scaleZoomOrigin[1]]
2017-04-10 16:28:22 +02:00
2022-05-06 09:41:22 +02:00
updateLines()
applyView()
}
}
function touchmove(e) {
2017-04-09 02:08:35 +02:00
2022-05-06 09:41:22 +02:00
updateLines()
if (e.touches.length == 1) {
2022-05-06 09:41:22 +02:00
mousemove(e.touches[0].clientX, e.touches[0].clientY)
} else if (e.touches.length == 2) {
const newPinchDistance = Math.sqrt(
Math.pow(e.touches[0].clientX - e.touches[1].clientX, 2)
+ Math.pow(e.touches[0].clientY - e.touches[1].clientY, 2)
2022-05-06 09:41:22 +02:00
)
2022-05-06 09:41:22 +02:00
zoom = initialPinchZoom * newPinchDistance / initialPinchDistance
2022-05-06 09:41:22 +02:00
const x = (e.touches[0].clientX + e.touches[1].clientX) / 2 - container.offsetLeft
const y = (e.touches[0].clientY + e.touches[1].clientY) / 2 - container.offsetTop
2022-05-06 09:41:22 +02:00
applyZoom(x, y, zoom)
}
}
function applyZoom(x, y, zoom) {
2022-05-06 09:41:22 +02:00
const deltaX = x - lastPosition[0]
const deltaY = y - lastPosition[1]
2022-05-06 09:41:22 +02:00
const pinchTranslateX = (x - container.clientWidth / 2 - deltaX)
const pinchTranslateY = (y - container.clientHeight / 2 - deltaY)
2022-05-06 09:41:22 +02:00
scaleZoomOrigin[0] = initialPinchZoomOrigin[0] + deltaX / zoom + pinchTranslateX / zoom - pinchTranslateX / initialPinchZoom
scaleZoomOrigin[1] = initialPinchZoomOrigin[1] + deltaY / zoom + pinchTranslateY / zoom - pinchTranslateY / initialPinchZoom
2022-05-06 09:41:22 +02:00
zoomOrigin[0] = scaleZoomOrigin[0] * zoom
zoomOrigin[1] = scaleZoomOrigin[1] * zoom
2022-05-06 09:41:22 +02:00
applyView()
updateLines()
}
window.addEventListener("mouseup", function (e) {
if (hovered.length > 0) {
container.style.cursor = "pointer"
} else if (drawing == true) {
container.style.cursor = "crosshair"
} else {
container.style.cursor = "default"
}
if (dragging) {
2022-05-06 09:41:22 +02:00
e.preventDefault()
}
2022-05-06 09:41:22 +02:00
mouseup(e.clientX, e.clientY)
})
window.addEventListener("touchend", touchend)
function mouseup(x, y) {
if (dragging) {
2022-05-06 09:41:22 +02:00
dragging = false
}
}
function touchend(e) {
if (e.touches.length == 0) {
2022-05-06 09:41:22 +02:00
mouseup()
} else if (e.touches.length == 1) {
2022-05-06 09:41:22 +02:00
initialPinchZoom = zoom
lastPosition = [e.touches[0].clientX, e.touches[0].clientY]
}
}
window.addEventListener("resize", function () {
2022-05-06 09:41:22 +02:00
//console.log(document.documentElement.clientWidth, document.documentElement.clientHeight)
2022-05-06 09:41:22 +02:00
applyView()
})
2022-04-12 14:18:53 +02:00
document.body.dataset.initDone = ''
}
function updateAtlasAll(atlas = atlasAll) {
2022-04-16 14:26:31 +02:00
for (const atlasIndex in atlas) {
if (Array.isArray(atlas[atlasIndex].path)) {
2022-04-16 14:26:31 +02:00
const currentPath = atlas[atlasIndex].path
atlas[atlasIndex].path = {}
atlas[atlasIndex].path[defaultPeriod] = currentPath
}
if (Array.isArray(atlas[atlasIndex].center)) {
2022-04-16 14:26:31 +02:00
const currentCenter = atlas[atlasIndex].center
atlas[atlasIndex].center = {}
atlas[atlasIndex].center[defaultPeriod] = currentCenter
}
if (atlas[atlasIndex].links) {
2022-04-16 14:26:31 +02:00
const currentLinks = atlas[atlasIndex].links
atlas[atlasIndex].links = {
website: [],
subreddit: [],
discord: [],
wiki: [],
...currentLinks
}
} else {
atlas[atlasIndex].links = {
website: [],
subreddit: [],
discord: [],
wiki: []
}
if (atlas[atlasIndex].website) atlas[atlasIndex].links.website = [atlas[atlasIndex].website]
if (atlas[atlasIndex].subreddit) atlas[atlasIndex].links.subreddit = atlas[atlasIndex].subreddit.split(',').map(subreddit => subreddit.trim().replace(/^\/r\//, ''))
delete atlas[atlasIndex].website
delete atlas[atlasIndex].subreddit
}
}
return atlas
}