2017-04-04 01:24:46 +02:00
|
|
|
/*
|
|
|
|
========================================================================
|
2022-04-09 09:45:47 +07:00
|
|
|
The 2022 /r/place Atlas
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 09:45:47 +07:00
|
|
|
An Atlas of Reddit's 2022 /r/place, with information to each
|
2017-04-04 01:24:46 +02:00
|
|
|
artwork of the canvas provided by the community.
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 09:45:47 +07:00
|
|
|
Copyright (c) 2017 Roland Rytz <roland@draemm.li>
|
|
|
|
Copyright (c) 2022 r/placeAtlas2 contributors
|
|
|
|
|
2017-04-04 01:24:46 +02:00
|
|
|
Licensed under the GNU Affero General Public License Version 3
|
2022-04-09 08:23:55 +07:00
|
|
|
https://place-atlas.stefanocoding.me/license.txt
|
2017-04-04 01:24:46 +02:00
|
|
|
========================================================================
|
|
|
|
*/
|
2022-04-09 09:45:47 +07:00
|
|
|
|
2017-04-08 05:57:12 +02:00
|
|
|
var linesCanvas = document.getElementById("linesCanvas");
|
|
|
|
var linesContext = linesCanvas.getContext("2d");
|
|
|
|
var hovered = [];
|
|
|
|
|
2017-04-10 16:28:22 +02:00
|
|
|
var previousZoomOrigin = [0, 0];
|
|
|
|
var previousScaleZoomOrigin = [0, 0];
|
|
|
|
|
2017-04-11 23:16:35 +02:00
|
|
|
var backgroundCanvas = document.createElement("canvas");
|
2022-04-03 11:08:52 +02:00
|
|
|
backgroundCanvas.width = 2000;
|
2022-04-04 22:39:49 +02:00
|
|
|
backgroundCanvas.height = 2000;
|
2017-04-11 23:16:35 +02:00
|
|
|
var backgroundContext = backgroundCanvas.getContext("2d");
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var wrapper = document.getElementById("wrapper");
|
|
|
|
|
|
|
|
var objectsContainer = document.getElementById("objectsList");
|
|
|
|
var closeObjectsListButton = document.getElementById("closeObjectsListButton");
|
|
|
|
|
|
|
|
var filterInput = document.getElementById("searchList");
|
|
|
|
|
|
|
|
var entriesList = document.getElementById("entriesList");
|
|
|
|
var hideListButton = document.getElementById("hideListButton");
|
|
|
|
var entriesListShown = false;
|
|
|
|
|
|
|
|
var sortedAtlas;
|
|
|
|
|
|
|
|
var entriesLimit = 50;
|
|
|
|
var entriesOffset = 0;
|
|
|
|
var moreEntriesButton = document.createElement("button");
|
|
|
|
moreEntriesButton.innerHTML = "Show "+entriesLimit+" more";
|
|
|
|
moreEntriesButton.id = "moreEntriesButton";
|
|
|
|
moreEntriesButton.onclick = function(){
|
|
|
|
buildObjectsList(null, null);
|
|
|
|
renderBackground();
|
|
|
|
render();
|
|
|
|
};
|
|
|
|
|
|
|
|
var defaultSort = "shuffle";
|
|
|
|
document.getElementById("sort").value = defaultSort;
|
|
|
|
|
|
|
|
var lastPos = [0, 0];
|
|
|
|
|
|
|
|
var fixed = false; // Fix hovered items in place, so that clicking on links is possible
|
|
|
|
|
|
|
|
if(document.documentElement.clientWidth > 2000){
|
|
|
|
entriesListShown = true;
|
|
|
|
wrapper.classList.remove('listHidden')
|
|
|
|
}
|
|
|
|
|
|
|
|
if(document.documentElement.clientWidth < 2000){
|
|
|
|
entriesListShown = false;
|
|
|
|
wrapper.classList.add('listHidden')
|
|
|
|
}
|
|
|
|
|
|
|
|
filterInput.addEventListener("input", function(e){
|
|
|
|
entriesOffset = 0;
|
|
|
|
entriesList.innerHTML = "";
|
|
|
|
entriesList.appendChild(moreEntriesButton);
|
|
|
|
|
|
|
|
if(this.value === ""){
|
|
|
|
document.getElementById("relevantOption").disabled = true;
|
|
|
|
sortedAtlas = atlas.concat();
|
|
|
|
buildObjectsList(null, null);
|
|
|
|
} else {
|
|
|
|
document.getElementById("relevantOption").disabled = false;
|
|
|
|
buildObjectsList(this.value.toLowerCase(), "relevant");
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
document.getElementById("sort").addEventListener("input", function(e){
|
|
|
|
entriesOffset = 0;
|
|
|
|
entriesList.innerHTML = "";
|
|
|
|
entriesList.appendChild(moreEntriesButton);
|
|
|
|
|
|
|
|
if(this.value != "relevant"){
|
|
|
|
defaultSort = this.value;
|
|
|
|
}
|
|
|
|
|
|
|
|
buildObjectsList(filterInput.value.toLowerCase(), this.value);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
hideListButton.addEventListener("click", function(e){
|
|
|
|
entriesListShown = !entriesListShown;
|
|
|
|
if(entriesListShown){
|
|
|
|
wrapper.classList.remove('listHidden')
|
|
|
|
} else {
|
|
|
|
wrapper.classList.add('listHidden')
|
|
|
|
}
|
2022-04-11 20:42:54 +07:00
|
|
|
updateHovering(e);
|
2022-04-09 16:27:14 +07:00
|
|
|
applyView();
|
|
|
|
render();
|
|
|
|
updateLines();
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
|
|
|
|
closeObjectsListButton.addEventListener("click", function(e){
|
|
|
|
hovered = [];
|
|
|
|
objectsContainer.innerHTML = "";
|
|
|
|
updateLines();
|
|
|
|
closeObjectsListButton.className = "hidden";
|
|
|
|
fixed = false;
|
|
|
|
render();
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
function toggleFixed(e, tapped){
|
|
|
|
if(!fixed && hovered.length == 0){
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
fixed = !fixed;
|
|
|
|
if(!fixed){
|
|
|
|
updateHovering(e, tapped);
|
|
|
|
render();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
window.addEventListener("resize", updateLines);
|
|
|
|
window.addEventListener("mousemove", updateLines);
|
|
|
|
window.addEventListener("dblClick", updateLines);
|
|
|
|
window.addEventListener("wheel", updateLines);
|
|
|
|
|
|
|
|
|
|
|
|
objectsContainer.addEventListener("scroll", function(e){
|
|
|
|
updateLines();
|
|
|
|
});
|
|
|
|
|
|
|
|
window.addEventListener("resize", function(){
|
|
|
|
//console.log(document.documentElement.clientWidth, document.documentElement.clientHeight);
|
|
|
|
|
|
|
|
var viewportWidth = document.documentElement.clientWidth;
|
|
|
|
|
|
|
|
if(document.documentElement.clientWidth > 2000 && viewportWidth <= 2000){
|
|
|
|
entriesListShown = true;
|
|
|
|
wrapper.className = wrapper.className.replace(/ listHidden/g, "");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(document.documentElement.clientWidth < 2000 && viewportWidth >= 2000){
|
|
|
|
entriesListShown = false;
|
|
|
|
wrapper.className += " listHidden";
|
|
|
|
}
|
2022-04-11 20:42:54 +07:00
|
|
|
updateHovering(e);
|
2022-04-09 16:27:14 +07:00
|
|
|
|
|
|
|
viewportWidth = document.documentElement.clientWidth;
|
|
|
|
|
|
|
|
applyView();
|
|
|
|
render();
|
|
|
|
updateLines();
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2017-04-08 05:57:12 +02:00
|
|
|
function updateLines(){
|
|
|
|
|
|
|
|
linesCanvas.width = linesCanvas.clientWidth;
|
|
|
|
linesCanvas.height = linesCanvas.clientHeight;
|
|
|
|
linesContext.lineCap = "round";
|
|
|
|
linesContext.lineWidth = Math.max(Math.min(zoom*1.5, 16*1.5), 6);
|
|
|
|
linesContext.strokeStyle = "#000000";
|
|
|
|
|
|
|
|
for(var i = 0; i < hovered.length; i++){
|
|
|
|
var element = hovered[i].element;
|
|
|
|
|
2017-04-09 02:08:35 +02:00
|
|
|
if(element.getBoundingClientRect().left != 0){
|
|
|
|
|
|
|
|
linesContext.beginPath();
|
|
|
|
//linesContext.moveTo(element.offsetLeft + element.clientWidth - 10, element.offsetTop + 20);
|
|
|
|
linesContext.moveTo(
|
|
|
|
element.getBoundingClientRect().left + document.documentElement.scrollLeft + element.clientWidth/2
|
|
|
|
,element.getBoundingClientRect().top + document.documentElement.scrollTop + 20
|
|
|
|
);
|
|
|
|
linesContext.lineTo(
|
|
|
|
~~(hovered[i].center[0]*zoom) + innerContainer.offsetLeft
|
|
|
|
,~~(hovered[i].center[1]*zoom) + innerContainer.offsetTop
|
|
|
|
);
|
|
|
|
linesContext.stroke();
|
|
|
|
|
|
|
|
}
|
2017-04-08 05:57:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
linesContext.lineWidth = Math.max(Math.min(zoom, 16), 4);
|
|
|
|
linesContext.strokeStyle = "#FFFFFF";
|
|
|
|
|
|
|
|
for(var i = 0; i < hovered.length; i++){
|
|
|
|
var element = hovered[i].element;
|
|
|
|
|
2017-04-09 02:08:35 +02:00
|
|
|
if(element.getBoundingClientRect().left != 0){
|
|
|
|
|
|
|
|
linesContext.beginPath();
|
|
|
|
linesContext.moveTo(
|
|
|
|
element.getBoundingClientRect().left + document.documentElement.scrollLeft + element.clientWidth/2
|
|
|
|
,element.getBoundingClientRect().top + document.documentElement.scrollTop + 20
|
|
|
|
);
|
|
|
|
linesContext.lineTo(
|
|
|
|
~~(hovered[i].center[0]*zoom) + innerContainer.offsetLeft
|
|
|
|
,~~(hovered[i].center[1]*zoom) + innerContainer.offsetTop
|
|
|
|
);
|
|
|
|
linesContext.stroke();
|
|
|
|
}
|
2017-04-08 05:57:12 +02:00
|
|
|
}
|
|
|
|
}
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2017-04-11 23:16:35 +02:00
|
|
|
function renderBackground(atlas){
|
|
|
|
|
|
|
|
backgroundContext.clearRect(0, 0, canvas.width, canvas.height);
|
|
|
|
|
|
|
|
//backgroundCanvas.width = 1000 * zoom;
|
|
|
|
//backgroundCanvas.height = 1000 * zoom;
|
|
|
|
|
|
|
|
//backgroundContext.lineWidth = zoom;
|
|
|
|
|
|
|
|
backgroundContext.fillStyle = "rgba(0, 0, 0, 0.6)";
|
|
|
|
backgroundContext.fillRect(0, 0, backgroundCanvas.width, backgroundCanvas.height);
|
|
|
|
|
|
|
|
for(var i = 0; i < atlas.length; i++){
|
|
|
|
|
|
|
|
var path = atlas[i].path;
|
|
|
|
|
|
|
|
backgroundContext.beginPath();
|
|
|
|
|
|
|
|
if(path[0]){
|
|
|
|
//backgroundContext.moveTo(path[0][0]*zoom, path[0][1]*zoom);
|
|
|
|
backgroundContext.moveTo(path[0][0], path[0][1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
for(var p = 1; p < path.length; p++){
|
|
|
|
//backgroundContext.lineTo(path[p][0]*zoom, path[p][1]*zoom);
|
|
|
|
backgroundContext.lineTo(path[p][0], path[p][1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
backgroundContext.closePath();
|
|
|
|
|
2022-04-07 21:06:16 -07:00
|
|
|
var bgStrokeStyle;
|
|
|
|
switch (atlas[i].diff) {
|
|
|
|
case "add":
|
|
|
|
bgStrokeStyle = "rgba(0, 255, 0, 1)";
|
|
|
|
backgroundContext.lineWidth = 2;
|
|
|
|
break;
|
|
|
|
case "edit":
|
|
|
|
bgStrokeStyle = "rgba(255, 255, 0, 1)";
|
|
|
|
backgroundContext.lineWidth = 2;
|
|
|
|
break;
|
2022-04-08 17:02:14 -03:00
|
|
|
case "delete":
|
|
|
|
bgStrokeStyle = "rgba(255, 0, 0, 1)";
|
|
|
|
backgroundContext.lineWidth = 2;
|
|
|
|
break;
|
2022-04-07 21:06:16 -07:00
|
|
|
default:
|
|
|
|
bgStrokeStyle = "rgba(255, 255, 255, 0.8)";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
backgroundContext.strokeStyle = bgStrokeStyle;
|
2017-04-11 23:16:35 +02:00
|
|
|
backgroundContext.stroke();
|
2022-04-07 21:06:16 -07:00
|
|
|
backgroundContext.lineWidth = 1;
|
2017-04-11 23:16:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
function buildObjectsList(filter, sort){
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(entriesList.contains(moreEntriesButton)){
|
|
|
|
entriesList.removeChild(moreEntriesButton);
|
2022-04-08 00:11:29 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(!sortedAtlas){
|
|
|
|
sortedAtlas = atlas.concat();
|
|
|
|
document.getElementById("atlasSize").innerHTML = "The Atlas contains "+sortedAtlas.length+" entries.";
|
|
|
|
}
|
2017-04-06 01:58:30 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(filter){
|
|
|
|
sortedAtlas = atlas.filter(function(value){
|
|
|
|
return (
|
|
|
|
value.name.toLowerCase().indexOf(filter) !== -1
|
|
|
|
|| value.description.toLowerCase().indexOf(filter) !== -1
|
|
|
|
|| value.subreddit && value.subreddit.toLowerCase().indexOf(filter) !== -1
|
2022-04-09 22:49:27 +07:00
|
|
|
|| value.id === filter
|
2022-04-09 16:27:14 +07:00
|
|
|
);
|
|
|
|
});
|
|
|
|
document.getElementById("atlasSize").innerHTML = "Found "+sortedAtlas.length+" entries.";
|
|
|
|
} else {
|
|
|
|
document.getElementById("atlasSize").innerHTML = "The Atlas contains "+sortedAtlas.length+" entries.";
|
2017-04-06 23:09:37 +02:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(sort === null){
|
|
|
|
sort = defaultSort;
|
2017-04-06 23:09:37 +02:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
renderBackground(sortedAtlas);
|
2017-04-06 23:09:37 +02:00
|
|
|
render();
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
document.getElementById("sort").value = sort;
|
2017-04-06 23:09:37 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
//console.log(sort);
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var sortFunction;
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
//console.log(sort);
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
switch(sort){
|
|
|
|
case "shuffle":
|
|
|
|
sortFunction = null;
|
|
|
|
if(entriesOffset == 0){
|
|
|
|
shuffle();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "alphaAsc":
|
|
|
|
sortFunction = function(a, b){
|
|
|
|
return a.name.toLowerCase().localeCompare(b.name.toLowerCase());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "alphaDesc":
|
|
|
|
sortFunction = function(a, b){
|
|
|
|
return b.name.toLowerCase().localeCompare(a.name.toLowerCase());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "newest":
|
|
|
|
sortFunction = function(a, b){
|
|
|
|
if (a.id > b.id) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (a.id < b.id) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
// a must be equal to b
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "oldest":
|
|
|
|
sortFunction = function(a, b){
|
|
|
|
if (a.id < b.id) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (a.id > b.id) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
// a must be equal to b
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "area":
|
|
|
|
sortFunction = function(a, b){
|
|
|
|
return calcPolygonArea(b.path) - calcPolygonArea(a.path);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "relevant":
|
|
|
|
sortFunction = function(a, b){
|
|
|
|
if(a.name.toLowerCase().indexOf(filter) !== -1 && b.name.toLowerCase().indexOf(filter) !== -1){
|
|
|
|
if (a.name.toLowerCase().indexOf(filter) < b.name.toLowerCase().indexOf(filter)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
else if (a.name.toLowerCase().indexOf(filter) > b.name.toLowerCase().indexOf(filter)) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return a.name.toLowerCase().localeCompare(b.name.toLowerCase());
|
|
|
|
}
|
|
|
|
} else if(a.name.toLowerCase().indexOf(filter) !== -1){
|
|
|
|
return -1;
|
|
|
|
} else if(b.name.toLowerCase().indexOf(filter) !== -1){
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
if (a.description.toLowerCase().indexOf(filter) < b.description.toLowerCase().indexOf(filter)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
else if (a.description.toLowerCase().indexOf(filter) > b.description.toLowerCase().indexOf(filter)) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return a.name.toLowerCase().localeCompare(b.name.toLowerCase());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(sortFunction){
|
|
|
|
sortedAtlas.sort(sortFunction);
|
|
|
|
}
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
for(var i = entriesOffset; i < entriesOffset+entriesLimit; i++){
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(i >= sortedAtlas.length){
|
|
|
|
break;
|
2017-04-09 21:45:08 +02:00
|
|
|
}
|
|
|
|
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var element = createInfoBlock(sortedAtlas[i]);
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
element.entry = sortedAtlas[i];
|
2017-04-09 02:08:35 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
element.addEventListener("mouseenter", function(e){
|
|
|
|
if(!fixed && !dragging){
|
|
|
|
objectsContainer.innerHTML = "";
|
|
|
|
|
|
|
|
previousZoomOrigin = [zoomOrigin[0], zoomOrigin[1]];
|
|
|
|
previousScaleZoomOrigin = [scaleZoomOrigin[0], scaleZoomOrigin[1]];
|
2017-04-10 12:25:04 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
applyView();
|
|
|
|
|
|
|
|
zoomOrigin = [
|
|
|
|
innerContainer.clientWidth/2 - this.entry.center[0]* zoom// + container.offsetLeft
|
|
|
|
,innerContainer.clientHeight/2 - this.entry.center[1]* zoom// + container.offsetTop
|
|
|
|
]
|
2017-04-06 01:58:30 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
scaleZoomOrigin = [
|
|
|
|
2000/2 - this.entry.center[0]
|
|
|
|
,2000/2 - this.entry.center[1]
|
|
|
|
]
|
2017-04-06 01:58:30 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
//console.log(zoomOrigin);
|
|
|
|
|
|
|
|
|
|
|
|
applyView();
|
|
|
|
hovered = [this.entry];
|
|
|
|
render();
|
|
|
|
hovered[0].element = this;
|
|
|
|
updateLines();
|
2017-04-06 01:58:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
element.addEventListener("click", function(e){
|
|
|
|
toggleFixed(e);
|
|
|
|
if(fixed){
|
|
|
|
previousZoomOrigin = [zoomOrigin[0], zoomOrigin[1]];
|
|
|
|
previousScaleZoomOrigin = [scaleZoomOrigin[0], scaleZoomOrigin[1]];
|
|
|
|
applyView();
|
|
|
|
}
|
|
|
|
if(document.documentElement.clientWidth < 500){
|
|
|
|
objectsContainer.innerHTML = "";
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
entriesListShown = false;
|
|
|
|
wrapper.className += " listHidden";
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
zoom = 4;
|
|
|
|
renderBackground(atlas);
|
|
|
|
applyView();
|
2022-04-11 20:42:54 +07:00
|
|
|
updateHovering(e);
|
2022-04-09 16:27:14 +07:00
|
|
|
|
|
|
|
zoomOrigin = [
|
|
|
|
innerContainer.clientWidth/2 - this.entry.center[0]* zoom// + container.offsetLeft
|
|
|
|
,innerContainer.clientHeight/2 - this.entry.center[1]* zoom// + container.offsetTop
|
|
|
|
]
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
scaleZoomOrigin = [
|
|
|
|
2000/2 - this.entry.center[0]
|
|
|
|
,2000/2 - this.entry.center[1]
|
|
|
|
]
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
previousZoomOrigin = [zoomOrigin[0], zoomOrigin[1]];
|
|
|
|
previousScaleZoomOrigin = [scaleZoomOrigin[0], scaleZoomOrigin[1]];
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
fixed = true;
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
hovered = [this.entry];
|
|
|
|
hovered[0].element = this;
|
|
|
|
|
|
|
|
applyView();
|
|
|
|
render();
|
|
|
|
updateLines();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
element.addEventListener("mouseleave", function(e){
|
|
|
|
if(!fixed && !dragging){
|
|
|
|
zoomOrigin = [previousScaleZoomOrigin[0]*zoom, previousScaleZoomOrigin[1]*zoom];
|
|
|
|
scaleZoomOrigin = [previousScaleZoomOrigin[0], previousScaleZoomOrigin[1]];
|
|
|
|
applyView();
|
|
|
|
hovered = [];
|
|
|
|
updateLines();
|
|
|
|
render();
|
|
|
|
}
|
|
|
|
});
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
entriesList.appendChild(element);
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
}
|
2017-04-09 02:08:35 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
entriesOffset += entriesLimit;
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(sortedAtlas.length > entriesOffset){
|
|
|
|
moreEntriesButton.innerHTML = "Show "+Math.min(entriesLimit, sortedAtlas.length - entriesOffset)+" more";
|
|
|
|
entriesList.appendChild(moreEntriesButton);
|
2017-04-04 01:24:46 +02:00
|
|
|
}
|
2022-04-09 16:27:14 +07:00
|
|
|
}
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
function shuffle(){
|
|
|
|
//console.log("shuffled atlas");
|
|
|
|
for (var i = sortedAtlas.length - 1; i > 0; i--) {
|
|
|
|
var j = Math.floor(Math.random() * (i + 1));
|
|
|
|
var temp = sortedAtlas[i];
|
|
|
|
sortedAtlas[i] = sortedAtlas[j];
|
|
|
|
sortedAtlas[j] = temp;
|
|
|
|
}
|
|
|
|
}
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
async function render(){
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
context.clearRect(0, 0, canvas.width, canvas.height);
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
//canvas.width = 1000*zoom;
|
|
|
|
//canvas.height = 1000*zoom;
|
|
|
|
|
|
|
|
context.globalCompositeOperation = "source-over";
|
|
|
|
context.clearRect(0, 0, canvas.width, canvas.height);
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(hovered.length > 0){
|
|
|
|
container.style.cursor = "pointer";
|
|
|
|
} else {
|
|
|
|
container.style.cursor = "default";
|
|
|
|
}
|
2017-04-09 21:45:08 +02:00
|
|
|
|
2017-04-09 22:22:06 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
for(var i = 0; i < hovered.length; i++){
|
2017-04-09 21:45:08 +02:00
|
|
|
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var path = hovered[i].path;
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
context.beginPath();
|
2017-04-06 19:44:09 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(path[0]){
|
|
|
|
//context.moveTo(path[0][0]*zoom, path[0][1]*zoom);
|
|
|
|
context.moveTo(path[0][0], path[0][1]);
|
2017-04-05 20:48:32 +02:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
for(var p = 1; p < path.length; p++){
|
|
|
|
//context.lineTo(path[p][0]*zoom, path[p][1]*zoom);
|
|
|
|
context.lineTo(path[p][0], path[p][1]);
|
2017-04-10 12:25:04 +02:00
|
|
|
}
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
context.closePath();
|
2017-04-10 15:16:34 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
context.globalCompositeOperation = "source-over";
|
2017-04-06 19:15:16 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
context.fillStyle = "rgba(0, 0, 0, 1)";
|
|
|
|
context.fill();
|
|
|
|
}
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
context.globalCompositeOperation = "source-out";
|
|
|
|
context.drawImage(backgroundCanvas, 0, 0);
|
|
|
|
|
|
|
|
if(hovered.length === 1 && hovered[0].path.length && hovered[0].overrideImage){
|
|
|
|
let undisputableHovered = hovered[0];
|
|
|
|
// Find the left-topmost point of all the paths
|
|
|
|
let entryPosition = getPositionOfEntry(undisputableHovered);
|
|
|
|
if(entryPosition){
|
|
|
|
const [startX, startY] = entryPosition;
|
|
|
|
let overrideImage = new Image();
|
|
|
|
const loadingPromise = new Promise((res, rej) => {
|
|
|
|
overrideImage.onerror = rej;
|
|
|
|
overrideImage.onload = res;
|
2017-04-05 20:48:32 +02:00
|
|
|
});
|
2022-04-09 16:27:14 +07:00
|
|
|
overrideImage.src = "imageOverrides/" + undisputableHovered.overrideImage;
|
|
|
|
try{
|
|
|
|
await loadingPromise;
|
|
|
|
context.globalCompositeOperation = "source-over";
|
|
|
|
context.drawImage(overrideImage, startX, startY);
|
|
|
|
}catch(ex){
|
|
|
|
console.log("Cannot override image.");
|
|
|
|
console.log(ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
for(var i = 0; i < hovered.length; i++){
|
2017-04-10 12:25:04 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var path = hovered[i].path;
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
context.beginPath();
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(path[0]){
|
|
|
|
//context.moveTo(path[0][0]*zoom, path[0][1]*zoom);
|
|
|
|
context.moveTo(path[0][0], path[0][1]);
|
2017-04-05 20:48:32 +02:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
for(var p = 1; p < path.length; p++){
|
|
|
|
//context.lineTo(path[p][0]*zoom, path[p][1]*zoom);
|
|
|
|
context.lineTo(path[p][0], path[p][1]);
|
2017-04-05 20:48:32 +02:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
context.closePath();
|
2017-04-11 23:16:35 +02:00
|
|
|
|
2017-04-05 20:48:32 +02:00
|
|
|
context.globalCompositeOperation = "source-over";
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var hoverStrokeStyle;
|
|
|
|
switch (hovered[i].diff) {
|
|
|
|
case "add":
|
|
|
|
hoverStrokeStyle = "rgba(0, 155, 0, 1)";
|
|
|
|
break;
|
|
|
|
case "edit":
|
|
|
|
hoverStrokeStyle = "rgba(155, 155, 0, 1)";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
hoverStrokeStyle = "rgba(0, 0, 0, 1)";
|
|
|
|
break;
|
2017-04-05 20:48:32 +02:00
|
|
|
}
|
2022-04-09 16:27:14 +07:00
|
|
|
context.strokeStyle = hoverStrokeStyle;
|
|
|
|
//context.lineWidth = zoom;
|
|
|
|
context.stroke();
|
|
|
|
}
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
}
|
2017-04-05 20:48:32 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
function updateHovering(e, tapped){
|
|
|
|
|
|
|
|
if(!dragging && (!fixed || tapped)){
|
|
|
|
var pos = [
|
|
|
|
(e.clientX - (container.clientWidth/2 - innerContainer.clientWidth/2 + zoomOrigin[0] + container.offsetLeft))/zoom
|
|
|
|
,(e.clientY - (container.clientHeight/2 - innerContainer.clientHeight/2 + zoomOrigin[1] + container.offsetTop))/zoom
|
|
|
|
];
|
|
|
|
var coords_p = document.getElementById("coords_p");
|
|
|
|
coords_p.innerText = Math.ceil(pos[0]) + ", " + Math.ceil(pos[1]);
|
|
|
|
|
|
|
|
if(pos[0] <= 2200 && pos[0] >= -100 && pos[0] <= 2200 && pos[0] >= -100){
|
|
|
|
var newHovered = [];
|
|
|
|
for(var i = 0; i < atlas.length; i++){
|
|
|
|
if(pointIsInPolygon(pos, atlas[i].path)){
|
|
|
|
newHovered.push(atlas[i]);
|
|
|
|
}
|
2017-04-04 01:24:46 +02:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var changed = false;
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(hovered.length == newHovered.length){
|
|
|
|
for(var i = 0; i < hovered.length; i++){
|
|
|
|
if(hovered[i].id != newHovered[i].id){
|
|
|
|
changed = true;
|
|
|
|
break;
|
|
|
|
}
|
2022-04-06 15:29:40 +02:00
|
|
|
}
|
2022-04-09 16:27:14 +07:00
|
|
|
} else {
|
|
|
|
changed = true;
|
2022-04-06 15:29:40 +02:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(changed){
|
|
|
|
hovered = newHovered.sort(function(a, b){
|
|
|
|
return calcPolygonArea(a.path) - calcPolygonArea(b.path);
|
|
|
|
});
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
objectsContainer.innerHTML = "";
|
2017-04-05 04:20:49 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
for(var i in hovered){
|
|
|
|
var element = createInfoBlock(hovered[i]);
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
objectsContainer.appendChild(element);
|
2017-04-05 04:20:49 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
hovered[i].element = element;
|
|
|
|
}
|
2017-04-05 04:20:49 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if(hovered.length > 0){
|
|
|
|
closeObjectsListButton.className = "";
|
|
|
|
} else {
|
|
|
|
closeObjectsListButton.className = "hidden";
|
|
|
|
}
|
2017-04-05 18:43:45 -04:00
|
|
|
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
render();
|
|
|
|
}
|
2017-04-04 01:24:46 +02:00
|
|
|
}
|
|
|
|
}
|
2022-04-09 16:27:14 +07:00
|
|
|
}
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 12:09:57 -07:00
|
|
|
window.addEventListener("hashchange", highlightEntryFromUrl);
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
function highlightEntryFromUrl(){
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var objectsContainer = document.getElementById("objectsList");
|
2017-04-10 12:25:04 +02:00
|
|
|
|
2022-04-10 19:50:39 -07:00
|
|
|
var id = window.location.hash.substring(1); //Remove hash prefix
|
2022-04-09 12:09:57 -07:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
var entries = atlas.filter(function(e){
|
|
|
|
return e.id === id;
|
2017-04-10 12:25:04 +02:00
|
|
|
});
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
if (entries.length === 1){
|
|
|
|
let entry = entries[0];
|
2017-04-04 01:24:46 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
document.title = entry.name + " on the 2022 /r/place Atlas";
|
|
|
|
|
|
|
|
var infoElement = createInfoBlock(entry);
|
|
|
|
objectsContainer.innerHTML = "";
|
|
|
|
objectsContainer.appendChild(infoElement);
|
2017-04-06 23:09:37 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
//console.log(entry.center[0]);
|
|
|
|
//console.log(entry.center[1]);
|
2017-04-10 12:25:04 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
zoom = 4;
|
|
|
|
renderBackground(atlas);
|
|
|
|
applyView();
|
|
|
|
|
|
|
|
zoomOrigin = [
|
|
|
|
innerContainer.clientWidth/2 - entry.center[0]* zoom// + container.offsetLeft
|
|
|
|
,innerContainer.clientHeight/2 - entry.center[1]* zoom// + container.offsetTop
|
|
|
|
];
|
2017-04-06 23:09:37 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
scaleZoomOrigin = [
|
|
|
|
2000/2 - entry.center[0]// + container.offsetLeft
|
|
|
|
,2000/2 - entry.center[1]// + container.offsetTop
|
|
|
|
];
|
2017-04-06 23:09:37 +02:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
//console.log(zoomOrigin);
|
2017-04-06 23:09:37 +02:00
|
|
|
|
|
|
|
applyView();
|
2022-04-09 16:27:14 +07:00
|
|
|
hovered = [entry];
|
2017-04-06 23:09:37 +02:00
|
|
|
render();
|
2022-04-09 16:27:14 +07:00
|
|
|
hovered[0].element = infoElement;
|
|
|
|
closeObjectsListButton.className = "";
|
2017-04-06 23:09:37 +02:00
|
|
|
updateLines();
|
2022-04-09 16:27:14 +07:00
|
|
|
fixed = true;
|
|
|
|
}
|
2017-04-05 18:43:45 -04:00
|
|
|
}
|
2022-04-09 10:41:40 +07:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
function initView(){
|
2022-04-09 10:41:40 +07:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
buildObjectsList(null, null);
|
2022-04-09 10:41:40 +07:00
|
|
|
renderBackground(atlas);
|
2022-04-09 16:27:14 +07:00
|
|
|
render();
|
2022-04-09 10:41:40 +07:00
|
|
|
|
|
|
|
timeCallback = (tempAtlas) => {
|
|
|
|
renderBackground(tempAtlas);
|
2022-04-09 16:27:14 +07:00
|
|
|
render();
|
2022-04-09 10:41:40 +07:00
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
// parse linked atlas entry id from link hash
|
|
|
|
/*if (window.location.hash.substring(3)){
|
|
|
|
zoom = 4;
|
|
|
|
applyView();
|
|
|
|
highlightEntryFromUrl();
|
|
|
|
}*/
|
2022-04-09 10:41:40 +07:00
|
|
|
|
|
|
|
applyView();
|
2022-04-09 16:27:14 +07:00
|
|
|
render();
|
|
|
|
updateLines();
|
2022-04-09 10:41:40 +07:00
|
|
|
|
2022-04-10 19:54:00 -07:00
|
|
|
if(window.location.hash){ // both "/" and just "/#" will be an empty hash string
|
2022-04-10 19:50:39 -07:00
|
|
|
highlightEntryFromUrl();
|
2022-04-09 16:27:14 +07:00
|
|
|
}
|
2022-04-09 10:41:40 +07:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
}
|
2022-04-09 10:41:40 +07:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
function initExplore(){
|
2022-04-09 10:41:40 +07:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
window.updateHovering = updateHovering
|
2022-04-09 10:41:40 +07:00
|
|
|
|
|
|
|
function updateHovering(e, tapped){
|
|
|
|
if(!dragging && (!fixed || tapped)){
|
|
|
|
var pos = [
|
|
|
|
(e.clientX - (container.clientWidth/2 - innerContainer.clientWidth/2 + zoomOrigin[0] + container.offsetLeft))/zoom
|
|
|
|
,(e.clientY - (container.clientHeight/2 - innerContainer.clientHeight/2 + zoomOrigin[1] + container.offsetTop))/zoom
|
|
|
|
];
|
|
|
|
var coords_p = document.getElementById("coords_p");
|
|
|
|
coords_p.innerText = Math.ceil(pos[0]) + ", " + Math.ceil(pos[1]);
|
2022-04-09 16:27:14 +07:00
|
|
|
|
2022-04-09 10:41:40 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
renderBackground(atlas);
|
|
|
|
|
|
|
|
applyView();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function initGlobal() {
|
|
|
|
container.addEventListener("mousemove", function(e){
|
|
|
|
if(e.sourceCapabilities){
|
|
|
|
if(!e.sourceCapabilities.firesTouchEvents){
|
|
|
|
updateHovering(e);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
updateHovering(e);
|
2022-04-09 10:41:40 +07:00
|
|
|
}
|
2022-04-09 16:27:14 +07:00
|
|
|
});
|
|
|
|
}
|
2022-04-09 10:41:40 +07:00
|
|
|
|
2022-04-09 16:27:14 +07:00
|
|
|
function initViewGlobal() {
|
2022-04-09 10:41:40 +07:00
|
|
|
container.addEventListener("mousedown", function(e){
|
|
|
|
lastPos = [
|
|
|
|
e.clientX
|
|
|
|
,e.clientY
|
|
|
|
];
|
|
|
|
});
|
2022-04-09 16:27:14 +07:00
|
|
|
|
2022-04-09 10:41:40 +07:00
|
|
|
container.addEventListener("touchstart", function(e){
|
|
|
|
if(e.touches.length == 1){
|
|
|
|
lastPos = [
|
|
|
|
e.touches[0].clientX
|
|
|
|
,e.touches[0].clientY
|
|
|
|
];
|
|
|
|
}
|
|
|
|
},{passive: true} );
|
2022-04-09 16:27:14 +07:00
|
|
|
|
2022-04-09 10:41:40 +07:00
|
|
|
container.addEventListener("mouseup", function(e){
|
|
|
|
if(Math.abs(lastPos[0] - e.clientX) + Math.abs(lastPos[1] - e.clientY) <= 4){
|
|
|
|
toggleFixed(e);
|
|
|
|
}
|
|
|
|
});
|
2022-04-09 16:27:14 +07:00
|
|
|
|
2022-04-09 10:41:40 +07:00
|
|
|
container.addEventListener("touchend", function(e){
|
|
|
|
e.preventDefault()
|
2022-04-09 16:27:14 +07:00
|
|
|
|
2022-04-09 10:41:40 +07:00
|
|
|
//console.log(e);
|
|
|
|
//console.log(e.changedTouches[0].clientX);
|
|
|
|
if(e.changedTouches.length == 1){
|
|
|
|
e = e.changedTouches[0];
|
|
|
|
//console.log(lastPos[0] - e.clientX);
|
|
|
|
if(Math.abs(lastPos[0] - e.clientX) + Math.abs(lastPos[1] - e.clientY) <= 4){
|
|
|
|
//console.log("Foo!!");
|
|
|
|
dragging = false;
|
|
|
|
fixed = false;
|
2022-04-09 16:27:14 +07:00
|
|
|
setTimeout(
|
|
|
|
function(){
|
|
|
|
updateHovering(e, true);
|
|
|
|
}
|
|
|
|
, 10);
|
2022-04-09 10:41:40 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2022-04-09 22:49:27 +07:00
|
|
|
}
|