/* ======================================================================== 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 r/placeAtlas2 contributors Licensed under the GNU Affero General Public License Version 3 https://place-atlas.stefanocoding.me/license.txt ======================================================================== */ const prodDomain = "place-atlas.stefanocoding.me" var innerContainer = document.getElementById("innerContainer"); var container = document.getElementById("container"); var canvas = document.getElementById("highlightCanvas"); var context = canvas.getContext("2d"); var zoom = 1; if(window.devicePixelRatio){ zoom = 1/window.devicePixelRatio; } var maxZoom = 128; var minZoom = 0.1; var zoomOrigin = [0, 0]; var scaleZoomOrigin = [0, 0]; var dragging = false; var lastPosition = [0, 0]; var viewportSize = [0, 0]; document.getElementById("entriesListDonate").addEventListener("click", function(e){ document.getElementById("donateOverlay").style.display = "flex"; }); document.getElementById("closeDonateButton").addEventListener("click", function(e){ document.getElementById("donateOverlay").style.display = "none"; }); function applyView(){ //console.log(zoomOrigin, scaleZoomOrigin); //console.log(scaleZoomOrigin[0]); scaleZoomOrigin[0] = Math.max(-1000, Math.min(1000, scaleZoomOrigin[0])); scaleZoomOrigin[1] = Math.max(-1000, Math.min(1000, scaleZoomOrigin[1])); zoomOrigin = [scaleZoomOrigin[0]*zoom, scaleZoomOrigin[1]*zoom]; innerContainer.style.height = (~~(zoom*2000))+"px"; innerContainer.style.width = (~~(zoom*2000))+"px"; 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"; } var atlas = null; if (document.location.host !== prodDomain) document.body.dataset.dev = "" init(); async function init(){ // For Reviewing Reddit Changes //let resp = await fetch("../tools/temp_atlas.json"); let resp = await fetch("./atlas.json"); atlas = await resp.json(); atlas.sort(function (a, b) { if (a.center[1] < b.center[1]) { return -1; } if (a.center[1] > b.center[1]) { return 1; } // a must be equal to b return 0; }); //TEMP FOR TIME TRAVEL atlasBackup = atlas; //console.log(document.documentElement.clientWidth, document.documentElement.clientHeight); zoomOrigin = [0, 0]; applyView(); var initialPinchDistance = 0; var initialPinchZoom = 0; var initialPinchZoomOrigin = [0, 0]; var desiredZoom; var zoomAnimationFrame; var mode = "view"; var args = window.location.search; if(args){ mode = args.split("mode=")[1]; if(mode){ mode = mode.split("&")[0]; } else { mode = "view"; } // Backwards compatibility for old links using "search" id arg if(args.includes("id=")){ let idHash = args.split("id=")[1].split("&")[0]; window.location.hash = idHash; let idArgMatch = new RegExp(`id=${idHash}&?`); // Patten for the id plus a following & if present window.location.search = window.location.search.substring(1).replace(idArgMatch, ""); } } document.body.dataset.mode = mode initGlobal() if (mode !== "draw") initViewGlobal() if(mode === "draw"){ initDraw(); } else if(mode === "about"){ window.location = "./about.html"; } else if(mode === "overlap"){ if(initOverlap){ initOverlap(); } } else if(mode.startsWith("diff")){ try { let liveResp = await fetch("https://place-atlas.stefanocoding.me/atlas.json"); let liveJson = await liveResp.json(); let liveAtlasReduced = liveJson.reduce(function(a, c) { a[c.id] = c; return a; },{}); // Mark added/edited entries atlas = atlas.map(function(entry) { if(liveAtlasReduced[entry.id] === undefined){ entry.diff = "add"; }else if(JSON.stringify(entry) !== JSON.stringify(liveAtlasReduced[entry.id])){ entry.diff = "edit"; } return entry; }); // Mark removed entries let atlasReduced = atlas.reduce(function(a, c) { a[c.id] = c; return a; },{}); let removedEntries = liveJson.filter(entry => atlasReduced[entry.id] === undefined ).map(entry => { entry.diff = "delete" return entry }) atlas.push(...removedEntries) if(mode.includes("only")){ atlas = atlas.filter(function(entry) { return typeof entry.diff == "string" }); } //TEMP FOR TIME TRAVEL atlasBackup = atlas; } catch (error) { console.warn("Diff mode failed to load, reverting to normal view.", error); } finally { if(initOverlap && mode.includes("overlap")){ initOverlap(); } else { initView(); } } } else if(mode === "explore"){ initExplore(); } else { initView(); } document.getElementById("loading").style.display = "none"; document.getElementById("zoomInButton").addEventListener("click", function(e){ /*if(zoomAnimationFrame){ window.cancelAnimationFrame(zoomAnimationFrame); }*/ var x = container.clientWidth/2; var y = container.clientHeight/2; initialPinchZoomOrigin = [ scaleZoomOrigin[0], scaleZoomOrigin[1] ]; initialPinchZoom = zoom; lastPosition = [x, y]; zoom = zoom * 2; zoom = Math.max(minZoom, Math.min(maxZoom, zoom)); applyZoom(x, y, zoom); }); document.getElementById("zoomOutButton").addEventListener("click", function(e){ /*if(zoomAnimationFrame){ window.cancelAnimationFrame(zoomAnimationFrame); }*/ var x = container.clientWidth/2; var y = container.clientHeight/2; initialPinchZoomOrigin = [ scaleZoomOrigin[0], scaleZoomOrigin[1] ]; initialPinchZoom = zoom; lastPosition = [x, y]; zoom = zoom / 2; zoom = Math.max(minZoom, Math.min(maxZoom, zoom)); applyZoom(x, y, zoom); }); document.getElementById("zoomResetButton").addEventListener("click", function(e){ zoom = 1; zoomOrigin = [0, 0]; scaleZoomOrigin = [0, 0]; updateLines(); applyView(); }); container.addEventListener("dblclick", function(e){ /*if(zoomAnimationFrame){ window.cancelAnimationFrame(zoomAnimationFrame); }*/ var x = e.clientX - container.offsetLeft; var y = e.clientY - container.offsetTop; initialPinchZoomOrigin = [ scaleZoomOrigin[0], scaleZoomOrigin[1] ]; initialPinchZoom = zoom; lastPosition = [x, y]; if(e.ctrlKey){ zoom = zoom / 2; } else { zoom = zoom * 2; } zoom = Math.max(minZoom, Math.min(maxZoom, zoom)); applyZoom(x, y, zoom); e.preventDefault(); }); container.addEventListener("wheel", function(e){ /*if(zoomAnimationFrame){ window.cancelAnimationFrame(zoomAnimationFrame); }*/ var x = e.clientX - container.offsetLeft; var y = e.clientY - container.offsetTop; initialPinchZoomOrigin = [ scaleZoomOrigin[0], scaleZoomOrigin[1] ]; initialPinchZoom = zoom; lastPosition = [x, y]; // 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 zoom -= e.deltaY * (0.001 * zoom); } else { if(e.deltaY > 0){ zoom = zoom / 2; } else if(e.deltaY < 0){ zoom = zoom * 2; } } zoom = Math.max(minZoom, Math.min(maxZoom, zoom)); applyZoom(x, y, zoom); e.preventDefault(); }, {passive: true}); /*function setDesiredZoom(x, y, target){ zoom = (zoom*2 + target)/3; //console.log(zoom); if(Math.abs(1 - zoom/target) <= 0.01){ zoom = target; } applyZoom(x, y, zoom); if(zoom != target){ zoomAnimationFrame = window.requestAnimationFrame(function(){ setDesiredZoom(x, y, target); }); } }*/ container.addEventListener("mousedown", function(e){ mousedown(e.clientX, e.clientY); e.preventDefault(); }); container.addEventListener("touchstart", function(e){ if(e.touches.length == 2){ e.preventDefault(); } touchstart(e); }, {passive: true}); function mousedown(x, y){ lastPosition = [x, y]; dragging = true; } function touchstart(e){ if(e.touches.length == 1){ 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) ); initialPinchZoom = zoom; initialPinchZoomOrigin = [ scaleZoomOrigin[0], scaleZoomOrigin[1] ]; mousedown( (e.touches[0].clientX + e.touches[1].clientX)/2, (e.touches[0].clientY + e.touches[1].clientY)/2 ); } } window.addEventListener("mousemove", function(e){ updateLines(); mousemove(e.clientX, e.clientY); if(dragging){ e.preventDefault(); } }); window.addEventListener("touchmove", function(e){ if(e.touches.length == 2 || e.scale > 1){ e.preventDefault(); } touchmove(e); }, {passive: false} ); function mousemove(x, y){ if(dragging){ var deltaX = x - lastPosition[0]; var deltaY = y - lastPosition[1]; lastPosition = [x, y]; zoomOrigin[0] += deltaX; zoomOrigin[1] += deltaY; scaleZoomOrigin[0] += deltaX/zoom; scaleZoomOrigin[1] += deltaY/zoom; previousZoomOrigin = [zoomOrigin[0], zoomOrigin[1]]; previousScaleZoomOrigin = [scaleZoomOrigin[0], scaleZoomOrigin[1]]; updateLines(); applyView(); } } function touchmove(e){ updateLines(); if(e.touches.length == 1){ mousemove(e.touches[0].clientX, e.touches[0].clientY); } else if(e.touches.length == 2){ var 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) ); zoom = initialPinchZoom * newPinchDistance / initialPinchDistance; var x = (e.touches[0].clientX + e.touches[1].clientX)/2 - container.offsetLeft; var y = (e.touches[0].clientY + e.touches[1].clientY)/2 - container.offsetTop; applyZoom(x, y, zoom); } } function applyZoom(x, y, zoom){ var deltaX = x - lastPosition[0]; var deltaY = y - lastPosition[1]; var pinchTranslateX = (x - container.clientWidth/2 - deltaX); var pinchTranslateY = (y - container.clientHeight/2 - deltaY); scaleZoomOrigin[0] = initialPinchZoomOrigin[0] + deltaX/zoom + pinchTranslateX/zoom - pinchTranslateX/initialPinchZoom; scaleZoomOrigin[1] = initialPinchZoomOrigin[1] + deltaY/zoom + pinchTranslateY/zoom - pinchTranslateY/initialPinchZoom; zoomOrigin[0] = scaleZoomOrigin[0]*zoom; zoomOrigin[1] = scaleZoomOrigin[1]*zoom; applyView(); updateLines(); } window.addEventListener("mouseup", function(e){ if(dragging){ e.preventDefault(); } mouseup(e.clientX, e.clientY); }); window.addEventListener("touchend", touchend); function mouseup(x, y){ if(dragging){ dragging = false; } } function touchend(e){ if(e.touches.length == 0){ mouseup(); } else if(e.touches.length == 1){ initialPinchZoom = zoom; lastPosition = [e.touches[0].clientX, e.touches[0].clientY]; } } window.addEventListener("resize", function(){ //console.log(document.documentElement.clientWidth, document.documentElement.clientHeight); applyView(); }); }