TBD
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

1988 lines
71 KiB

(function(window, document, appState, undefined){
//console.log("app.js");
const svgXMLNS = "http://www.w3.org/2000/svg";
const xmlnsXMLNS = 'http://www.w3.org/2000/xmlns/';
const xmlSpaceXMLNS = 'http://www.w3.org/XML/1998/namespace';
const xmlXlinkXMLNS = 'http://www.w3.org/1999/xlink';
const textPostMaxLines = 7;
const maxInt = 2147483647;
const polaroidTextMaxLines = 2;
const polaroidTextVerticalFudge = -25;
const polaroidRandomizeButtonFudge = [0,50];
const textPostDefaultMaxWidth = 400;
const textPostLineSpacing = 1.7;
const textPostFontSizePixels = 26;
const textPostWidthEstimationFudgeFactor = 1;
const textPostTextLineLimitFudgeFactor = 1.2;
const textPostPadding = [30,40];
const textPostRandomTranslate = [0,0];
const textPostRandomBezier = [10,30];
const textLineRandomTranslate = [22,4];
const textLineRandomBezier = [10,1];
const imageUploadTypeIconSize = "100px";
const imagePostPreviewCanvasMaxDimension = 1024;
const polaroidBorderTop = 55;
const polaroidBorderLeft = 100;
const polaroidBorderBottom = 229;
const polaroidBorderRight = 100;
const polaroidInsetTop = 51;
const polaroidInsetLeft = 51;
const polaroidInsetBottom = 215;
const polaroidInsetRight = 55;
const polaroidBorder = [polaroidBorderTop, polaroidBorderLeft, polaroidBorderBottom, polaroidBorderRight];
const polaroidBorderThickness = 0.5;
const backgroundImageThumbnailBPGDataURI = "data:image/bpg;base64,QlBH+yAAggCCAAADkkdARAHBcYMSAAABJgGvCEgmMEMZQLWl4Nfa62/sLtgsBdsYs7rqKFN/lk16+VmPanaO50tVLXKpXtXv6IA7IxMS734VZGRKbk61L/22O6fvdDR0+k3y//DdKcEfpVlQP4qb4PoS6ggQPNA/g5VIECpY1Jdtl9g0UOT8lgV3KuR9qUZzJqDC5ZOyPQbnIxcU5o7DF0TrOaMggkH07mxy/vqZm22SnfkGQzx7IQ1XtUYysFJ5Lnn1F6Ba48HNUX5dhjFLcuzkIor/DSCH/+9mn1J6iNFoWTR+nOknAvwATHL0XCy6qWTetwEWgJSqX/OaPFaSxMANQ8vTNZ9jC/VtNcUq64tBv9ZLas6PONnZxkHtiGOkCZrNjz+6Q+ongsjbrnLYDsvVw/bArLjG2LeAnVpXfnbTyXycBcXvTvGkKSe5GRzzUMbOKlFp6z1U2sRk8l0onFq4Y/UcvzPPIvRnpazMW3315d2+YXyE6g/HlzeTPYCu18q/qJ5WtVnrsEYw6sS1BkfweYiUf1AQUD9Ee6zA8uiT2dxTbgqMfq//1PbihCo5wAjGCYWVxIuefG1zNfwKJfIq1hV3sy2utIVxjWT+LWfXrFV3KA/75/MI2km2P8RKc7jZWBInW7IjrrswSMY7JZJq63+CxslP5PBSK1C9HPMnSAvV";
const jpegHeaderBase64 = "/9j/2wCEADUlKC8oITUvKy88OTU/UIVXUElJUKN1e2GFwarLyL6qurfV8P//1eL/5re6////////////zv////////////8BOTw8UEZQnVdXnf/cutz////////////////////////////////////////////////////////////////////AABEIACAAIAMBIgACEQEDEQH/xAGiAAABBQEBAQEBAQAAAAAAAAAAAQIDBAUGBwgJCgsQAAIBAwMCBAMFBQQEAAABfQECAwAEEQUSITFBBhNRYQcicRQygZGhCCNCscEVUtHwJDNicoIJChYXGBkaJSYnKCkqNDU2Nzg5OkNERUZHSElKU1RVVldYWVpjZGVmZ2hpanN0dXZ3eHl6g4SFhoeIiYqSk5SVlpeYmZqio6Slpqeoqaqys7S1tre4ubrCw8TFxsfIycrS09TV1tfY2drh4uPk5ebn6Onq8fLz9PX29/j5+gEAAwEBAQEBAQEBAQAAAAAAAAECAwQFBgcICQoLEQACAQIEBAMEBwUEBAABAncAAQIDEQQFITEGEkFRB2FxEyIygQgUQpGhscEJIzNS8BVictEKFiQ04SXxFxgZGiYnKCkqNTY3ODk6Q0RFRkdISUpTVFVWV1hZWmNkZWZnaGlqc3R1dnd4eXqCg4SFhoeIiYqSk5SVlpeYmZqio6Slpqeoqaqys7S1tre4ubrCw8TFxsfIycrS09TV1tfY2dri4+Tl5ufo6ery8/T19vf4+fr/2gAMAwEAAhEDEQA/A";
// see static/test.png for the source of this image.
const testBPGImageDataURI = "data:image/bpg;base64,QlBH+3AAICAAA5JCUAOSQlBECcF1gRIAAAABRAHBdYESAAABJgmusLkxB6WjYAAAASYBr1i28siTPC6H7on+4sC2h4bAl2edvY7Djuh2ZwC8S9WS+Y2qtILgTWZX/g4H4h8zG3PPOQWOs4ZscvkyuvyG3aMqtKbtKdCfCFJbNQ==";
const backgroundImageSize = [1000, 656];
const backgroundScale = 2;
const postScale = 0.5;
const edgePanThreshold = window.innerWidth > window.innerHeight ? window.innerHeight * 0.22 : window.innerWidth * 0.22;
const edgePanPower = 2;
const edgePanSpeed = 1500;
const absorbVelocityRate = 7;
const velocityDamping = 5;
const postsTilesPollingDurationSeconds = 60;
const postTypes = {
text: 1,
image: 2,
video: 3,
}
let loader;
let newPostElement;
let inviteElement;
let inviteIcon;
let postIcon;
let inviteLabel;
let postLabel;
let openNewPost;
let openInvite;
let textPostPreviewContainer;
let imagePostIsTransparent;
let imageFile;
let imageFileEXIFOrientation;
let imagePreviewCanvas;
let newPostRandomSeed = new Date().getTime();
let newPostFontIndex = Math.round(mulberry32(newPostRandomSeed)()*Number.MAX_SAFE_INTEGER);
let lastMaxLines = 7;
let lastDefaultTextValue = "...";
let dragIcon;
let postButton;
let viewX = 0;
let viewY = 0;
let velocityX = 0;
let velocityY = 0;
let positioningPost;
let positioningPostX = 0;
let positioningPostY = 0;
let edgePanX = 0;
let edgePanY = 0;
let dragPanning = false;
let edgePanning = false;
let wallContainer;
let postsContainer;
let loadedPosts = {};
let loadedTiles = {};
let colliderQuadrants = {};
let collidersAllowedOverlapPx = 7;
let currentlyZoomedInPost;
let currentlyZooming = false;
let firstTilePosition = null;
let lastInteractionTimestamp = new Date().getTime();
let currentlyLoadingPosts = false;
let firstPostsLoaded = false;
let supportsBPG;
window.addEventListener('DOMContentLoaded', () => {
//console.log("DOMContentLoaded");
detectBPGSupport();
});
window.onBPGSupportDetected = (supported) => {
//console.log("window.onBPGSupportDetected() was called");
if(supportsBPG !== undefined || supported == undefined) {
return;
}
supportsBPG = supported;
window.localStorage['supportsBPG'] = String(supportsBPG);
let registrationPage;
let lastTextPostValue;
loader = document.getElementById("progress-container");
wallContainer = createElement(document.body, "div", {
class: "wall",
style: {
width: `${Math.round(backgroundImageSize[0]*backgroundScale*2)}px`,
height: `${Math.round(backgroundImageSize[1]*backgroundScale*2)}px`,
}
});
postsContainer = createElement(document.body, "div", {
class: "posts pointer-events-none",
});
createTiledBackground(wallContainer);
// https://png-pixel.com clear 1x1 image
dragIcon = createElement(
document.body,
"img",
{src: "data:image/gif;base64,R0lGODlhAQABAIAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw== "}
);
setupGenericDrag(
wallContainer,
[wallContainer],
(dx, dy) => {
viewX -= dx;
viewY -= dy;
dragPanning = true;
},
() => {
dragPanning = false;
}
);
let lastTimestamp;
function animationFrame(timestamp) {
if(lastTimestamp) {
update( (timestamp-lastTimestamp)*0.001 )
}
lastTimestamp = timestamp;
window.requestAnimationFrame(animationFrame);
}
window.requestAnimationFrame(animationFrame);
let lastViewX = viewX;
let lastViewY = viewY;
let collisionDetectionFrameCounter = 0;
function update(deltaTime) {
if(positioningPost) {
const dx = edgePanX*deltaTime*edgePanSpeed;
const dy = edgePanY*deltaTime*edgePanSpeed;
viewX += dx;
viewY += dy;
positioningPostX += dx;
positioningPostY += dy;
positioningPost.style.transform = `translate(${positioningPostX}px, ${positioningPostY}px) scale(${postScale}) `;
// only run collisionDetect() every 4 frames.
if(postButton && (collisionDetectionFrameCounter % 4) != collisionDetectionFrameCounter) {
const postContainer = positioningPost.querySelector(".post-container");
if(collisionDetect(createCollider(positioningPost.firstChild))) {
postContainer.classList.remove("legal");
postContainer.classList.add("illegal");
postButton.disabled = true;
} else {
postContainer.classList.add("legal");
postContainer.classList.remove("illegal");
postButton.removeAttribute('disabled');
}
collisionDetectionFrameCounter = 0;
}
collisionDetectionFrameCounter++;
}
velocityX = lerp(velocityX, 0, velocityDamping*deltaTime);
velocityY = lerp(velocityY, 0, velocityDamping*deltaTime);
if(edgePanX == 0 && edgePanY == 0 && !dragPanning) {
viewX += velocityX*deltaTime;
viewY += velocityY*deltaTime;
if(edgePanning) {
positioningPostX += velocityX*deltaTime;
positioningPostY += velocityY*deltaTime;
}
} else {
if(deltaTime != 0) {
const realVelocityX = (viewX-lastViewX)/deltaTime;
const realVelocityY = (viewY-lastViewY)/deltaTime;
lastViewX = viewX;
lastViewY = viewY;
velocityX = lerp(velocityX, realVelocityX, absorbVelocityRate*deltaTime);
velocityY = lerp(velocityY, realVelocityY, absorbVelocityRate*deltaTime);
}
}
let wallX = -viewX;
let wallY = -viewY;
while(wallX > 0) {
wallX -= backgroundImageSize[0]*backgroundScale;
}
while(wallY > 0) {
wallY -= backgroundImageSize[1]*backgroundScale;
}
wallX = wallX % Math.round(backgroundImageSize[0]*backgroundScale);
wallY = wallY % Math.round(backgroundImageSize[1]*backgroundScale);
wallContainer.style.transform = `translate(${wallX}px, ${wallY}px)`;
postsContainer.style.transform = `translate(${Math.round(-viewX)}px, ${Math.round(-viewY)}px)`;
}
if( (!appState.IsAuthor && appState.Invited) ) {
//console.log("(!appState.IsAuthor && appState.Invited)");
loader.style.display = 'none';
registrationPage = createElement(
document.body,
"div",
{class: "fullscreen-flying registration vertical align-center justify-center"}
);
createElement(
registrationPage,
"h2",
{class: "line-height-150 text-center margin-0 padding-2"},
"You're invited to post on our collage wall! 😀"
);
const form = createElement(registrationPage, "div", {class: "horizontal align-center padding-2"});
createElement(form, "label", {for: "registration-display-name", class: "text-center"}, "Screen Name: ");
const input = createElement(form, "input", {id: "registration-display-name", type: "text"});
createElement(form, "span", null, " (Optional)");
const button = createElement(registrationPage, "button", null, "Continue");
button.onclick = () => {
window.location = `/register?displayName=${input.value.trim()}`;
};
return
}
const overlay = createElement(document.body, "div", {class: "fullscreen-flying pointer-events-none"});
if(appState.TemporaryError) {
createElement(
overlay,
"div",
{class: "temporary-error text-center padding-1 line-height-150"},
appState.TemporaryError
);
} else if(!appState.IsAuthor) {
createElement(
overlay,
"div",
{class: "registration text-center padding-1 line-height-150"},
`Anyone may view this collage wall, but you must be invited to post.
Someone who already has access can invite you, or you can visit the
Cyberia Computer Club's booth #R69 to request an invite!`
);
} else {
createElement(overlay, "div", {class: "instance-id"}, window.graffitiAppState.InstanceId);
newPostElement = createElement(overlay, "div", {class: "post-button material-style-button centered-flex"});
postIcon = createElement(newPostElement, "img", {
src: "static/spray-paint.webp",
alt: "create new post"
});
postLabel = createElement(createElement(newPostElement, "label", null), "span", null, "Post");
inviteElement = createElement(overlay, "div", {class: "invite-button material-style-button centered-flex"});
inviteIcon = createElement(inviteElement, "img", {
src: "static/qr-code.webp",
alt: "invite another user"
});
inviteLabel = createElement(createElement(inviteElement, "label", null), "span", null, "Invite");
openInvite = () => {
inviteElement.onclick = null;
inviteElement.classList.add("invite-expand");
inviteIcon.style.display = "none";
inviteLabel.style.display = "none";
const inviteBody = createElement(document.body, "div", {class: "vertical align-center hidden"});
const animateInviteSize = () => {
inviteElement.style.width = `380px`;
inviteElement.style.height = `${inviteBody.clientHeight}px`;
};
xhr("GET", `invite`, null, (statusCode, responseText) => {
if (statusCode == 200) {
const inviteCode = responseText;
const inviteLink = `${window.location.protocol}//${window.location.host}/join/${inviteCode}`;
const inviteLinkHex = string2Hex(inviteLink);
createElement(inviteBody, "img", {
src: `/qr/${inviteLinkHex}`,
alt: "qr code",
class: "qr-code"
});
const inviteCodeContainer = createElement(inviteBody, "div", {class: "horizontal justify-center margin-2"});
const inviteLinkInput = createElement(
createElement(inviteCodeContainer, "div", {class: "invite-code link"}),
"input",
{value: inviteLink, disabled: true}
);
const copyIconContainer = createElement(inviteCodeContainer, "div", {class: "invite-code icon"});
createElement(copyIconContainer, "img", {src: "static/copy-icon.png", alt: "copy invite link"});
const toastContainer = createElement(inviteCodeContainer, "div", {class: "position-absolute"});
copyIconContainer.onclick = () => {
inviteLinkInput.removeAttribute("disabled");
inviteLinkInput.select();
inviteLinkInput.setSelectionRange(0, inviteLink.length);
const result = document.execCommand("copy");
inviteLinkInput.setSelectionRange(0, 0);
inviteLinkInput.disabled = true;
const toast = createElement(
toastContainer,
"div",
{class: "invite-link-copied-toast"},
result ? "Link copied to clipboard!" : "Copy to clipboard fail :("
);
setTimeout(() => {
toast.style.top = "4.5em";
toast.style.opacity = "1";
}, 10);
setTimeout(() => {
toast.style.transitionTimingFunction = 'ease-in-out';
toast.style.top = "5.5em";
toast.style.opacity = "0";
setTimeout(() => {
toastContainer.removeChild(toast);
}, 1000);
}, 4000);
};
createElement(inviteBody, 'button', {
class: "basic-button",
onclick: (e) => {
inviteElement.classList.remove("invite-expand");
inviteElement.style.removeProperty("width");
inviteElement.style.removeProperty("height");
inviteElement.removeChild(inviteBody);
setTimeout(() => {
inviteElement.onclick = openInvite;
inviteIcon.style.display = "block";
inviteLabel.style.display = "block";
}, 500);
},
style: {
marginTop: "0",
marginBottom: "1rem"
}
}, "Close");
animateInviteSize();
} else {
throw new Error("error attempting to load invite code: " + responseText);
}
});
setTimeout(() => {
inviteElement.appendChild(inviteBody);
inviteBody.classList.remove("hidden");
}, 500);
};
inviteElement.onclick = openInvite;
openNewPost = () => {
newPostElement.onclick = null;
newPostElement.classList.add("post-expand");
postIcon.style.display = "none";
postLabel.style.display = "none";
let continueButton;
const postForm = createElement(document.body, "div", {class: "vertical align-center hidden"});
const contentPreviewArea = createElement(postForm, "div", {class: "vertical align-center"});
const animateFormSize = () => {
newPostElement.style.width = `380px`;
newPostElement.style.height = `${postForm.clientHeight}px`;
};
const validateForm = () => {
if(continueButton) {
continueButton.disabled = Boolean(textInput?.value?.trim?.() == "" && !imageFile);
}
};
let textInput;
const reRenderTextPostPreview = (maxLines, defaultTextValue) => {
lastMaxLines = maxLines;
lastDefaultTextValue = defaultTextValue;
clearElementContents(textPostPreviewContainer);
const isPolaroid = imageFile && !imagePostIsTransparent;
const maxWidth = isPolaroid ? Number(imagePreviewCanvas.style.width.replace("px", "")) : textPostDefaultMaxWidth;
if(maxWidth == NaN) {
throw new Error("reRenderTextPostPreview(): imagePreviewCanvas must have a defined width in pixels.")
}
const textPostSVGContainer = createElement(textPostPreviewContainer, "div", {
class: "vertical align-center",
style: {
width: `${maxWidth}px`,
height: `${Math.round(315*(maxLines/textPostMaxLines))}px`,
}
});
validateForm();
createText(textPostSVGContainer, textInput.value || defaultTextValue || "", maxLines, newPostRandomSeed, newPostFontIndex, maxWidth);
if(isPolaroid) {
const svg = textPostSVGContainer.querySelector('svg');
if(svg) {
const elementHeightPx = Number(imagePreviewCanvas.style.height.replace("px", ""));
const svgWidth = svg.clientWidth;
const widthFactor = svgWidth/textPostDefaultMaxWidth;
textPostPreviewContainer.style.top = `${Math.round(elementHeightPx+polaroidTextVerticalFudge*widthFactor*polaroidBorderThickness)}px`;
}
}
animateFormSize();
};
const createRandomizeButton = (parent, maxLines) => {
const polaroid = parent.querySelector(".polaroid");
const container = createElement(parent, "div", {class: "text-post-randomize-button-container"});
const randomizeButton = createElement(
container,
"div",
{
class: "material-style-button text-post-randomize-button centered-flex",
onclick: () => {
newPostFontIndex ++;
newPostRandomSeed = new Date().getTime();
reRenderTextPostPreview(lastMaxLines, lastDefaultTextValue);
if(polaroid) {
const randomBorder = mulberry32(newPostRandomSeed)() > 0.5 ? '1' : '2';
polaroid.style.borderImageSource = `url('static/polaroid-border-${randomBorder}.webp')`;
}
}
}
);
createElement(randomizeButton, "img", {
src: "static/dice.webp",
alt: "randomize",
});
return container;
}
const switchToTextPost = () => {
clearElementContents(contentPreviewArea);
displayUploadOptions();
imagePreviewCanvas = null;
imageFile = null;
const postPreview = createElement(contentPreviewArea, "div", {class: "text-post-container"});
textPostPreviewContainer = createElement(postPreview, "div", {class: "vertical align-center"});
reRenderTextPostPreview(textPostMaxLines, "...");
createRandomizeButton(postPreview);
validateForm();
};
const onImageSelected = (e) => {
clearElementContents(contentPreviewArea);
imageFile = e.target.files[0];
imageFileEXIFOrientation = 0;
// getJPEGEXIFOrientationFromFile(imageFile, (orientation) => {
// if (orientation > 0) {
// imageFileEXIFOrientation = orientation
// }
// });
createImagePreview(contentPreviewArea, e.target.files[0]).then((canvas) => {
imagePreviewCanvas = canvas;
const elementHeightPx = Number(canvas.style.height.replace("px", ""));
if(elementHeightPx == NaN) {
throw new Error("onImageSelected(): Invalid argument canvas: element must have a defined size in pixels.")
}
imagePostIsTransparent = canvasIsTransparent(canvas);
if(imagePostIsTransparent) {
// Transparent images don't get the polaroid border -- just handle it like a text post
// and slam the image in right above the text.
canvas.style.display = 'block';
const postPreview = createElement(contentPreviewArea, "div", {
class: "text-post-container vertical align-center",
style: { paddingTop: '1em' },
});
postPreview.appendChild(canvas);
textPostPreviewContainer = createElement(postPreview, "div", {class: "vertical align-center"});
reRenderTextPostPreview(textPostMaxLines, "...");
createRandomizeButton(contentPreviewArea);
} else {
// opaque images (photos) get the polaroid border with theh text overlayed on the bottom of the border
const randomBorder = mulberry32(newPostRandomSeed)() > 0.5 ? '1' : '2';
const polaroid = createPolaroid(contentPreviewArea, canvas, `polaroid-border-${randomBorder}.webp`);
textPostPreviewContainer = createElement(
createElement(polaroid, "div", {class: "position-absolute "}),
"div", { style: { position: "relative" } }
);
reRenderTextPostPreview(polaroidTextMaxLines, "");
const randomizeButton = createRandomizeButton(polaroid);
randomizeButton.style.left = `${Math.round(polaroidRandomizeButtonFudge[0]*polaroidBorderThickness)}px`;
randomizeButton.style.top = `${Math.round(elementHeightPx+(polaroidBorderBottom+polaroidRandomizeButtonFudge[1])*polaroidBorderThickness)}px`;
}
validateForm();
}).catch(console.error);
};
const onRotateImage = () => {
};
const displayUploadOptions = () => {
const uploadOptions = createElement(contentPreviewArea, "div", {class: "singleselect-bar horizontal margin-2 upload-options"});
const uploadImageLabel = createElement(
uploadOptions,
"label",
{
for: "upload-image"
}
);
createElement(
uploadImageLabel,
"img",
{src: `static/image.svg`, width: imageUploadTypeIconSize, height: imageUploadTypeIconSize}
);
createElement(
uploadOptions,
"input",
{
accept: "image/jpeg, image/png",
id: "upload-image",
type: "file",
class: "display-none",
onchange: onImageSelected,
}
);
const captureVideoButton = createElement(
uploadOptions,
"label",
// {
// "aria-role": "button",
// "onclick": () => {
// uploadOptions.style.display = "none";
// },
// }
);
createElement(
captureVideoButton,
"img",
{
src: `static/video.png`,
width: imageUploadTypeIconSize,
height: imageUploadTypeIconSize,
style: {opacity: "0.4"}
}
);
createElement(
createElement( captureVideoButton, "div", {class: "position-absolute"} ),
"span",
{class: "upload-video-is-not-supported-yet"},
"video is not supported yet"
);
};
const onTextInputChanged = debounce(() => reRenderTextPostPreview(lastMaxLines, lastDefaultTextValue), 250);
const checkIfTextInputChanged = () => {
if(textInput.value != lastTextPostValue) {
onTextInputChanged();
lastTextPostValue = textInput.value;
}
};
textInput = createElement(postForm, 'textarea', {
placeholder: "type your message here...",
onchange: checkIfTextInputChanged,
onkeyup: checkIfTextInputChanged,
});
textInput.focus();
switchToTextPost();
const buttonContainer = createElement(postForm, "div", {
class: "horizontal space-between",
style: {width: "360px"},
});
createElement(buttonContainer, 'button', {
class: "basic-button",
onclick: (e) => {
newPostElement.classList.remove("post-expand");
newPostElement.style.removeProperty("width");
newPostElement.style.removeProperty("height");
newPostElement.removeChild(postForm);
setTimeout(() => {
newPostElement.onclick = openNewPost;
postIcon.style.display = "block";
postLabel.style.display = "block";
}, 500);
},
}, "Cancel");
continueButton = createElement(buttonContainer, 'button', {
class: "basic-button",
onclick: (e) => {
newPostElement.classList.remove("post-expand");
newPostElement.classList.add("post-button-enter-placement-mode");
newPostElement.style.removeProperty("width");
newPostElement.style.removeProperty("height");
newPostElement.removeChild(postForm);
enterPostPositioningMode(textInput.value);
},
}, "Continue...");
validateForm();
animateFormSize();
setTimeout(() => {
newPostElement.appendChild(postForm);
postForm.classList.remove("hidden");
}, 500);
}
newPostElement.onclick = openNewPost;
}
const centerOfViewX = Math.round(viewX+window.innerWidth*0.5);
const centerOfViewY = Math.round(viewY+window.innerHeight*0.5);
firstTilePosition = [centerOfViewX, centerOfViewY];
reloadPostsTile(firstTilePosition, "0,0");
setInterval(() => {
if(currentlyLoadingPosts) {
return;
}
const predictedFutureX = Math.round(viewX+velocityX+window.innerWidth*0.5);
const predictedFutureY = Math.round(viewY+velocityY+window.innerHeight*0.5);
const tile = getPostsTile(predictedFutureX, predictedFutureY);
if(!loadedTiles[tile.tileKey] || (lastInteractionTimestamp - loadedTiles[tile.tileKey])*0.001 > postsTilesPollingDurationSeconds) {
reloadPostsTile(tile.position, tile.tileKey);
}
}, 500);
};
function getPostsTile(x,y) {
const tileCoordX = Math.round((x-firstTilePosition[0])/window.graffitiAppState.PostsTileSize);
const tileCoordY = Math.round((y-firstTilePosition[1])/window.graffitiAppState.PostsTileSize);
return {
position: [
firstTilePosition[0]+tileCoordX*window.graffitiAppState.PostsTileSize,
firstTilePosition[1]+tileCoordY*window.graffitiAppState.PostsTileSize
],
tileKey: `${tileCoordX},${tileCoordY}`
};
}
function reloadPostsTile(position, tileKey) {
currentlyLoadingPosts = true;
return new Promise(resolve => {
xhr("GET", `posts?x=${position[0]}&y=${position[1]}&cacheBust=${new Date().getTime()}`, null, (statusCode, responseText) => {
lastInteractionTimestamp = new Date().getTime();
loadedTiles[tileKey] = new Date().getTime();
onPostsLoaded(statusCode, responseText);
currentlyLoadingPosts = false;
resolve();
});
});
}
function onPostsLoaded(statusCode, postsJSON) {
//console.log("onPostsLoaded");
if(statusCode == 200) {
if(!firstPostsLoaded) {
firstPostsLoaded = true;
loader.style.display = 'none';
}
const posts = JSON.parse(postsJSON);
if(posts.length) {
let spawningPostsInterval;
spawningPostsInterval = setInterval(() => {
let post = posts.shift();
while(posts.length && (loadedPosts[post.id] || document.getElementById(post.id))) {
post = posts.shift();
}
loadedPosts[post.id] = true;
createPost(post, false);
if(!posts.length) {
clearInterval(spawningPostsInterval)
}
}, 100);
}
// posts.forEach(post => {
// });
} else {
throw new Error("error attempting to load posts: " + postsJSON);
}
}
function enterPostPositioningMode(textContent) {
postsContainer.draggable = false;
// force re-download of nearby posts -- try to avoid submitting posts that overlap a post someone else just now created.
const centerOfViewX = Math.round(viewX+window.innerWidth*0.5);
const centerOfViewY = Math.round(viewY+window.innerHeight*0.5);
const tile = getPostsTile(centerOfViewX, centerOfViewY);
loader.style.display = "block";
reloadPostsTile(tile.position, tile.tileKey).then(() => {
loader.style.display = "none";
let width = 0;
let height = 0;
if(imageFile) {
width = Number(imagePreviewCanvas.style.width.replace("px", ""));
height = Number(imagePreviewCanvas.style.height.replace("px", ""));
if(width == NaN || height == NaN) {
throw new Error("enterPostPositioningMode(): imagePreviewCanvas must have a defined size in pixels.");
}
}
positioningPostX = viewX+(window.innerWidth*0.5);
positioningPostY = viewY+(window.innerHeight*0.5);
const metadata = {
x: Math.round(positioningPostX),
y: Math.round(positioningPostY),
width: Math.round(width),
height: Math.round(height),
angle: Math.round(0),
randomSeed: newPostRandomSeed,
fontIndex: newPostFontIndex,
textContent: textContent,
//imageFileEXIFOrientation: imageFileEXIFOrientation,
type: imageFile ? postTypes.image : postTypes.text,
transparentImage: imagePostIsTransparent,
};
positioningPost = createPost(metadata, true);
const testCollider = createCollider(positioningPost.firstChild);
if(collisionDetect(testCollider)) {
const distanceIncrementPixels = 50;
const directionsToTry = getRandomPermutation([[1,0],[1,1],[0,1],[-1,1],[-1,0],[-1,-1],[0,-1],[1,-1]]);
let foundASpot = false;
let i = 1;
while(i < 200 && !foundASpot) {
for(const direction of directionsToTry) {
const dx = direction[0]*i*distanceIncrementPixels;
const dy = direction[1]*i*distanceIncrementPixels;
if(!collisionDetect(cloneAndTranslateCollider(testCollider, dx, dy))) {
foundASpot = true;
positioningPostX += dx;
positioningPostY += dy;
viewX += dx;
viewY += dy;
break;
}
}
i++;
};
}
const postContainer = positioningPost.querySelector(".post-container");
postContainer.classList.add("legal");
postContainer.style.opacity = "0.5";
setupGenericDrag(
positioningPost,
[positioningPost, wallContainer],
(dx, dy, x, y) => {
positioningPostX += dx;
positioningPostY += dy;
let edgePanMagnitude = [
1-clamp01(x/edgePanThreshold),
clamp01((x-(window.innerWidth-edgePanThreshold))/edgePanThreshold),
1-clamp01(y/edgePanThreshold),
clamp01((y-(window.innerHeight-edgePanThreshold))/edgePanThreshold),
].reduce((agg, x) => x > agg ? x : agg, 0);
edgePanMagnitude = Math.pow(edgePanMagnitude, edgePanPower);
edgePanX = x - window.innerWidth*0.5;
edgePanY = y - window.innerHeight*0.5;
const magnitude = Math.sqrt(edgePanX*edgePanX + edgePanY*edgePanY);
edgePanX = (edgePanX/magnitude)*edgePanMagnitude;
edgePanY = (edgePanY/magnitude)*edgePanMagnitude;
edgePanning = true;
},
() => {
edgePanX = 0;
edgePanY = 0;
edgePanning = false;
}
);
let overlay;
const cleanup = () => {
loader.style.display = 'none';
document.body.removeChild(overlay);
newPostElement.classList.remove("post-button-enter-placement-mode");
setTimeout(() => {
postIcon.style.display = "block";
postLabel.style.display = "block";
newPostElement.onclick = openNewPost;
}, 500);
postContainer.classList.remove("legal");
postContainer.classList.remove("illegal");
postContainer.style.opacity = "1";
// // TODO figure out how to make it draggable (to pan) AND clickable (to zoom in).
// postContainer.draggable = false;
};
const postButtonClicked = () => {
loader.style.display = 'block';
clearElementContents(overlay);
overlay.className = "fullscreen-flying modal-background centered-flex";
metadata.x = Math.round(positioningPostX);
metadata.y = Math.round(positioningPostY);
metadata.collider = createCollider(positioningPost.firstChild);
registerCollider(metadata.collider);
setupDragOnPost(positioningPost);
positioningPost.style.transition ="transform";
positioningPost.style.transitionTimingFunction = "cubic-bezier(.24,-0.4,.58,1)";
positioningPost.style.transitionDuration = "0.5s";
positioningPost = null;
xhr("POST", "upload-meta", metadata, (status, responseText) => {
loader.style.display = 'none';
if(status != 200) {
cleanup();
throw new Error(`POST /upload-meta: server returned HTTP ${status}: ${responseText}`)
}
let uploadPromise = Promise.resolve();
if(metadata.type == postTypes.image) {
const postId = responseText;
uploadPromise = new Promise((resolve, reject) => {
const formData = new FormData();
formData.append('file', imageFile);
const progressWindow = createElement(
createElement(overlay, "div", {class: "centered-flex"}),
"div",
{class: "progress-window"}
);
createElement(progressWindow, "p", null, "Uploading...");
const progressBar = createElement(
createElement(progressWindow, "div", {class: "progress-bar-holder"}),
"div",
{class: "progress-bar"}
);
let progressBarInterval = setInterval(() => {
xhr("GET", `progress/${postId}`, null, (status, responseText) => {
if(status == 200) {
progressBar.style.width = `calc(${responseText}% - 6px)`;
} else {
progressWindow.querySelector("p").textContent = `HTTP ${status}: ${responseText}`;
}
});
}, 333);
xhr("POST", `upload/${postId}`, formData, (status, responseText) => {
cleanup();
clearInterval(progressBarInterval);
if(status != 200) {
throw new Error(`POST /upload-meta: server returned HTTP ${status}: ${responseText}`)
}
});
});
} else {
cleanup();
}
});
};
overlay = createElement(document.body, "div", {class: "fullscreen-flying pointer-events-none vertical space-between align-center"});
createElement(
overlay,
"div",
{class: "registration text-center padding-1 line-height-150"},
'You may touch and drag to position your post, then click POST! to publish it.'
);
postButton = createElement( overlay, 'button', {
class: "basic-button submit-post-button padding-2",
onclick: postButtonClicked,
},
"POST!"
);
});
}
function createCollider(e) {
const bounds = e.getBoundingClientRect();
let toReturn;
if(bounds.width > bounds.height) {
const r = bounds.height*0.5;
toReturn = {
r: Math.round(r),
x1: Math.round(bounds.x+viewX+r),
y1: Math.round(bounds.y+viewY+r),
x2: Math.round(bounds.x+viewX+(bounds.width-r)),
y2: Math.round(bounds.y+viewY+r),
}
} else {
const r = bounds.width*0.5;
toReturn = {
r: Math.round(r),
x1: Math.round(bounds.x+viewX+r),
y1: Math.round(bounds.y+viewY+r),
x2: Math.round(bounds.x+viewX+r),
y2: Math.round(bounds.y+viewY+(bounds.height-r)),
}
}
generateBoundsOnCollider(toReturn);
return toReturn;
}
function generateBoundsOnCollider(c) {
const bounds = {
minX: (c.x1 < c.x2 ? c.x1 : c.x2)-c.r,
maxX: (c.x1 > c.x2 ? c.x1 : c.x2)+c.r,
minY: (c.y1 < c.y2 ? c.y1 : c.y2)-c.r,
maxY: (c.y1 > c.y2 ? c.y1 : c.y2)+c.r,
};
// TODO remove this quick hack later
bounds.minX += collidersAllowedOverlapPx
bounds.maxX -= collidersAllowedOverlapPx
bounds.minY += collidersAllowedOverlapPx
bounds.maxY -= collidersAllowedOverlapPx
c.bounds = bounds;
}
function cloneAndTranslateCollider(collider, x, y) {
const c = {...collider};
c.bounds = {...collider.bounds};
c.x1 += Math.round(x);
c.y1 += Math.round(y);
c.x2 += Math.round(x);
c.y2 += Math.round(y);
c.bounds.minX += x;
c.bounds.minY += y;
c.bounds.maxX += x;
c.bounds.maxY += y;
return c;
}
function registerCollider(c) {
generateBoundsOnCollider(c);
// all colliders are capsule colliders (2 points and a radius defines a capsule)
// pre-calculate the vector and normalized vector along the axis of the capsule.
c.vx = c.x1 - c.x2;
c.vy = c.y1 - c.y2;
const n = normalize(c.vx, c.vy);
c.nx = n[0];
c.ny = n[1];
getQuadrantsIntersectingBounds(c.bounds).forEach(k => {
if(!colliderQuadrants[k]) {
colliderQuadrants[k] = [];
}
colliderQuadrants[k].push(c);
});
}
function getQuadrantsIntersectingBounds(bounds) {
const quadrants = {};
quadrants[`${Math.round(bounds.minX/1000)},${Math.round(bounds.minY/1000)}`] = true;
quadrants[`${Math.round(bounds.maxX/1000)},${Math.round(bounds.minY/1000)}`] = true;
quadrants[`${Math.round(bounds.maxX/1000)},${Math.round(bounds.maxY/1000)}`] = true;
quadrants[`${Math.round(bounds.minX/1000)},${Math.round(bounds.maxY/1000)}`] = true;
return Object.keys(quadrants);
}
function collisionDetect(collider) {
const quadrants = getQuadrantsIntersectingBounds(collider.bounds);
for(const quadrant of quadrants) {
const collidersInQuadrant = colliderQuadrants[quadrant] || []
for(const other of collidersInQuadrant) {
const rect1 = collider.bounds;
const rect2 = other.bounds;
// https://developer.mozilla.org/en-US/docs/Games/Techniques/2D_collision_detection#axis-aligned_bounding_box
if (rect1.minX < rect2.maxX && rect1.maxX > rect2.minX && rect1.minY < rect2.maxY && rect1.maxY > rect2.minY) {
// TODO these are supposed to be capsule colliders but I don't have the time for this rn.
// const [ix, iy] = intersectTwoLines(collider, other);
// const [ix, iy] = intersectTwoLines(collider, other);
// // if the lines don't intersect, the must be parallel
// if(ix == null && iy == null) {
// transpose
// }
return true
}
}
}
return false;
}
function createPost(metadata, isPreview) {
if(document.getElementById(metadata.id)) {
return
}
if(!isPreview) {
registerCollider(metadata.collider);
}
const isTextPost = (!isPreview && metadata.type == postTypes.text) || (isPreview && !imageFile);
const anchorPoint = createElement(postsContainer, "div", {
class: "position-absolute",
id: metadata.id,
style:{
transform: `translate(${metadata.x}px, ${metadata.y}px) scale(${postScale})`,
}
});
// we can't do this yet if its a preview -- the css transitions will destroy the drag and drop physics.
// so we have to do it later when we exit drag and drop placement mode
if(!isPreview) {
anchorPoint.style.transition ="transform";
anchorPoint.style.transitionTimingFunction = "cubic-bezier(.24,-0.4,.58,1)";
anchorPoint.style.transitionDuration = "0.5s";
}
const postContainer = createElement(anchorPoint, "div", {
class: `position-absolute vertical align-center post-container ${isTextPost ? '' : 'opacity-0'} ${(isTextPost || metadata.transparentImage) ? 'no-polaroid' : ''}`,
});
let text;
if( isTextPost ) {
text = createText(postContainer, metadata.textContent, textPostMaxLines, metadata.randomSeed, metadata.fontIndex);
} else {
let thumbnailJPEGBase64 = metadata.thumbnailJPEGBase64;
if(thumbnailJPEGBase64 && !thumbnailJPEGBase64.startsWith("/9j")) {
thumbnailJPEGBase64 = `${jpegHeaderBase64}${thumbnailJPEGBase64}`;
}
let imageElement;
if(isPreview) {
imageElement = imagePreviewCanvas;
postContainer.classList.remove("opacity-0");
} else {
imageElement = createElement(postContainer, "div", {
class: "image-container",
style:{width: `${metadata.width}px`, height: `${metadata.height}px`}
});
const thumbnail = createElement(imageElement, "img", {
class: "image-load-blur",
style:{
width: `${metadata.width}px`,
height: `${metadata.height}px`,
position: 'absolute',
},
src: `data:image/jpeg;base64,${thumbnailJPEGBase64}`,
onload: () => postContainer.classList.remove("opacity-0"),
});
imageElement.dataset.filename = metadata.filename;
imageElement.dataset.width = metadata.width;
imageElement.dataset.height = metadata.height;
loadImageIntoContainer(
imageElement,
`${metadata.filename}-small`,
metadata.width,
metadata.height,
"image-load-blur",
(newImage) => {
imageElement.removeChild(thumbnail);
thumbnail.remove();
newImage.dataset.size = "small";
setTimeout(() => newImage.classList.add("resolve-0"), 10);
setTimeout(() => newImage.classList.add("resolve"), 510);
}
);
}
if(metadata.transparentImage) {
postContainer.appendChild(imageElement);
if(metadata.textContent) {
text = createText(postContainer, metadata.textContent, textPostMaxLines, metadata.randomSeed, metadata.fontIndex, textPostDefaultMaxWidth);
}
} else {
const randomBorder = mulberry32(metadata.RandomSeed)() > 0.5 ? '1' : '2';
const polaroid = createPolaroid(postContainer, imageElement, `polaroid-border-${randomBorder}.webp`);
if(metadata.textContent) {
const widthFactor = metadata.width/textPostDefaultMaxWidth;
const textContainer = createElement(
polaroid,
"div", {
class: "vertical align-center position-absolute",
style: {
position: "relative",
top: `${Math.round(metadata.height+polaroidTextVerticalFudge*widthFactor*polaroidBorderThickness)}px`,
width: `${metadata.width}px`,
height: `${Math.round(315*(polaroidTextMaxLines/textPostMaxLines))}px`
}
}
);
text = createText(textContainer, metadata.textContent, polaroidTextMaxLines, metadata.randomSeed, metadata.fontIndex, metadata.width);
}
}
}
postContainer.style.left = `${postContainer.clientWidth*-0.5}px`;
postContainer.style.top = `${postContainer.clientHeight*-0.5}px`;
if(!isPreview) {
setupDragOnPost(anchorPoint);
}
return anchorPoint;
}
function setupDragOnPost(postElement) {
setupGenericDrag(
postElement,
[postElement, postsContainer],
(dx, dy) => {
viewX -= dx;
viewY -= dy;
dragPanning = true;
},
() => {
dragPanning = false;
},
() => {
zoomInPost(postElement);
}
);
}
function zoomInPost(postElement) {
if(currentlyZooming) {
return
}
currentlyZooming = true;
// we zoom in the post as long as one is not already zoomed in.
if(currentlyZoomedInPost != null) {
zoomOutPost();
return
}
currentlyZoomedInPost = postElement;
const bounds = postElement.firstChild.getBoundingClientRect();
postElement.dataset.originalTransform = postElement.style.transform;
const match = /translate\(([0-9.-]+)px, ([0-9.-]+)px\)/.exec(postElement.style.transform);
const originalX = Number(match[1]);
const originalY = Number(match[2]);
const newX = originalX + ((window.innerWidth *0.5)-((bounds.left+bounds.right )*0.5));
const newY = originalY + ((window.innerHeight*0.5)-((bounds.top +bounds.bottom)*0.5));
const match2 = /scale\(([0-9.]+)\)/.exec(postElement.style.transform);
const currentScale = Number(match2[1]);
const newScale1 = window.innerWidth/bounds.width;
const newScale2 = window.innerHeight/bounds.height;
const newScale = currentScale * (newScale1 < newScale2 ? newScale1 : newScale2) * 0.95;
postElement.style.zIndex = 100;
postElement.style.transform = `translate(${newX}px, ${newY}px) scale(${newScale})`;
const noPolaroid = postElement.querySelector(".no-polaroid");
if(noPolaroid) {
noPolaroid.classList.add("focused");
}
// posts that the user just created won't have .image-container
// they are already high-res so we don't need to download the high res version.
const imageContainer = postElement.querySelector(".image-container");
if(imageContainer) {
const existingImage = imageContainer.querySelector("canvas,img");
loadImageIntoContainer(
imageContainer,
`${imageContainer.dataset.filename}-large`,
imageContainer.dataset.width,
imageContainer.dataset.height,
"display-none",
(newImage) => {
if(newImage) {
newImage.dataset.size = "large";
}
if(newImage && existingImage) {
existingImage.classList.add("display-none");
newImage.classList.remove("display-none");
}
currentlyZooming = false;
}
);
} else {
currentlyZooming = false;
}
}
function zoomOutPost() {
if(currentlyZooming) {
return
}
currentlyZooming = true;
if(currentlyZoomedInPost) {
currentlyZoomedInPost.style.transform = currentlyZoomedInPost.dataset.originalTransform;
const noPolaroid = currentlyZoomedInPost.querySelector(".no-polaroid");
if(noPolaroid) {
noPolaroid.classList.remove("focused");
}
const prevZoomedInPost = currentlyZoomedInPost;
const imageContainer = currentlyZoomedInPost.querySelector(".image-container");
setTimeout(() => {
prevZoomedInPost.style.removeProperty("z-index");
// posts that the user just created won't have .image-container
// they were already high-res so we don't need clean up the high res version.
if(imageContainer) {
const images = Array.from(imageContainer.querySelectorAll("canvas,img"));
const imagesBySize = {};
images.forEach(e => imagesBySize[e.dataset.size] = e);
if (imagesBySize.small && imagesBySize.large) {
imagesBySize.small.classList.remove("display-none");
imageContainer.removeChild(imagesBySize.large);
imagesBySize.large.remove();
}
}
currentlyZooming = false;
}, 500);
currentlyZoomedInPost = null;
}
}
function loadImageIntoContainer(container, filename, width, height, className, onload) {
if(supportsBPG) {
const result = window.createBPGElement(
container,
`user-content/${filename}.bpg`,
className,
`${width}px`,
`${height}px`
);
result.promise.then(onload);
} else {
let mainImg;
mainImg = createElement(container, "img", {
class: className,
style: {
width: `${width}px`,
height: `${height}px`,
position: 'absolute',
},
src: `user-content/${filename}.jpeg`,
onload: () => onload(mainImg)
});
}
}
function createPolaroid(parent, canvas, border) {
const elementWidthPx = Number(canvas.style.width.replace("px", ""));
const elementHeightPx = Number(canvas.style.height.replace("px", ""));
if(elementWidthPx == NaN || elementHeightPx == NaN) {
throw new Error("createPolaroid(): Invalid argument canvas: element must have a defined size in pixels.");
}
const polaroidContainer = createElement(parent, "div", {
class: "polaroid-container",
style: {
width: `${elementWidthPx}px`,
height: `${elementHeightPx}px`,
}
});
polaroidContainer.style.marginTop =`${Math.round(polaroidInsetTop*polaroidBorderThickness)}px`;
polaroidContainer.style.marginBottom = `${Math.round(polaroidInsetBottom*polaroidBorderThickness)}px`;
polaroidContainer.appendChild(canvas);
canvas.classList.add("position-absolute");
createElement(createElement(polaroidContainer, "div", { class: "position-absolute" }), "div", {
class: "polaroid",
style: {
borderImageSource: `url('static/${border}')`,
borderImageSlice: `${polaroidBorder.join(" ")}`,
borderTop: `${Math.round(polaroidBorderTop*polaroidBorderThickness)}px solid`,
borderLeft: `${Math.round(polaroidBorderLeft*polaroidBorderThickness)}px solid`,
borderBottom: `${Math.round(polaroidBorderBottom*polaroidBorderThickness)}px solid`,
borderRight: `${Math.round(polaroidBorderRight*polaroidBorderThickness)}px solid`,
top: `${Math.round(-polaroidInsetTop*polaroidBorderThickness)}px`,
left: `${Math.round(-polaroidInsetLeft*polaroidBorderThickness)}px`,
bottom: `${Math.round(-polaroidInsetBottom*polaroidBorderThickness)}px`,
right: `${Math.round(-polaroidInsetRight*polaroidBorderThickness)}px`,
width: `${Math.round(elementWidthPx+(polaroidInsetLeft+polaroidInsetRight)*polaroidBorderThickness)}px`,
height: `${Math.round(elementHeightPx+(polaroidInsetTop+polaroidInsetBottom)*polaroidBorderThickness)}px`,
}
});
canvas.style.display = 'block';
return polaroidContainer;
}
function createTiledBackground(parent) {
if(!supportsBPG) {
createElement(parent, "div", {
class: "jpeg-background",
style: {
width: `${backgroundImageSize[0]*backgroundScale*2}px`,
height: `${backgroundImageSize[1]*backgroundScale*2}px`,
}
});
return
}
let loadedFullRes = false;
const makeTile = (canvas, x,y, fullRes) => {
if (fullRes != loadedFullRes) {
return
}
const tile = createElement(
parent,
"canvas",
{
width: canvas.width,
height: canvas.height,
class: "position-absolute image-load-blur pointer-events-none",
style: {
left: `${x*backgroundImageSize[0]*backgroundScale}px`,
top: `${y*backgroundImageSize[1]*backgroundScale}px`,
width: `${backgroundImageSize[0]*backgroundScale}px`,
height: `${backgroundImageSize[1]*backgroundScale}px`
}}
);
const originalContext = canvas.getContext("2d");
const newContext = tile.getContext("2d");
newContext.putImageData(originalContext.getImageData(0,0,canvas.width,canvas.height), 0, 0);
if(fullRes) {
setTimeout(() => tile.classList.add("resolve-0"), 10);
setTimeout(() => tile.classList.add("resolve"), 510);
}
};
// first create the thumbnail background -- this should happen pretty much instantly
window.createBPGElement(
parent,
backgroundImageThumbnailBPGDataURI,
"position-absolute image-load-blur pointer-events-none",
`${backgroundImageSize[0]*backgroundScale}px`,
`${backgroundImageSize[1]*backgroundScale}px`
).promise.then((originalTile) => {
makeTile(originalTile, 0, 1, false);
makeTile(originalTile, 1, 1, false);
makeTile(originalTile, 1, 0, false);
});
// then download the actual background image, and when it loads, swap out the thumbnail for the real one.
window.createBPGElement(
document.body,
"static/blue-wall-tile-1000.bpg",
"position-absolute image-load-blur pointer-events-none display-none",
`${backgroundImageSize[0]*backgroundScale}px`,
`${backgroundImageSize[1]*backgroundScale}px`
).promise.then((originalTile) => {
clearElementContents(parent);
loadedFullRes = true;
originalTile.classList.remove("display-none");
parent.appendChild(originalTile);
setTimeout(() => originalTile.classList.add("resolve-0"), 10);
setTimeout(() => originalTile.classList.add("resolve"), 510);
makeTile(originalTile, 0, 1, true);
makeTile(originalTile, 1, 1, true);
makeTile(originalTile, 1, 0, true);
});
}
function createImagePreview(parent, imgFile) {
return new Promise((resolve, reject) => {
const img = createElement(document.body, "img", {
class: "display-none",
src: window.URL.createObjectURL(imgFile),
onload: () => {
let newWidth = img.naturalWidth;
let newHeight = img.naturalHeight;
if (newWidth > newHeight) {
if (img.naturalWidth > imagePostPreviewCanvasMaxDimension) {
newHeight *= imagePostPreviewCanvasMaxDimension / img.naturalWidth;
newWidth = imagePostPreviewCanvasMaxDimension;
}
} else if (img.naturalHeight > imagePostPreviewCanvasMaxDimension) {
newWidth *= imagePostPreviewCanvasMaxDimension / img.naturalHeight;
newHeight = imagePostPreviewCanvasMaxDimension;
}
const canvasWidth = newWidth > 380 ? Math.round(newWidth*(380/newWidth)) : Math.round(newWidth);
const canvasHeight = newWidth > 380 ? Math.round(newHeight*(380/newWidth)) : Math.round(newHeight);
const canvas = createElement(
parent,
"canvas",
{
width: Math.round(newWidth),
height: Math.round(newHeight),
style: {
display: "none",
width: `${canvasWidth}px`,
height: `${canvasHeight}px`,
}
}
);
const context2d = canvas.getContext("2d");
context2d.drawImage(img, 0, 0, Math.round(newWidth), Math.round(newHeight));
document.body.removeChild(img);
resolve(canvas);
}
});
});
}
function canvasIsTransparent (canvas) {
var context2d = canvas.getContext("2d");
const numberOfOpaquePixels = [
[0,0],
[Math.round(canvas.width*0.5),0],
[canvas.width-1,0],
[canvas.width-1,Math.round(canvas.height*0.5)],
[canvas.width-1,canvas.height-1],
[Math.round(canvas.width*0.5),canvas.height-1],
[0,canvas.height-1],
[0,Math.round(canvas.height*0.5)],
].map(([x,y]) => (context2d.getImageData(x, y, 1, 1).data[3] == 255) ? 1 : 0).reduce((a,v) => a+v);
return numberOfOpaquePixels < 7;
}
function setupGenericDrag(element, backgroundElements, onDragged, onDragEnd, onClick) {
element.draggable = true;
let lastDragX = 0;
let lastDragY = 0;
let dragging = false;
const dragStart = (e) => {
if(currentlyZoomedInPost != null) {
zoomOutPost();
}
dragging = true;
lastDragX = e.touches?.[0]?.clientX || e.clientX;
lastDragY = e.touches?.[0]?.clientY || e.clientY;
e.dataTransfer.setDragImage(dragIcon, 0, 0);
};
const dragOver = (e) => {
if(!dragging) {
return;
}
lastInteractionTimestamp = new Date().getTime();
const x = e.touches?.[0]?.clientX || e.clientX;
const y = e.touches?.[0]?.clientY || e.clientY;
onDragged?.(x-lastDragX, y-lastDragY, x,y);
lastDragX = x;
lastDragY = y;
};
element.addEventListener('dragstart', dragStart);
element.addEventListener('touchstart', dragStart);
backgroundElements.forEach(element => {
element.addEventListener('dragover', dragOver);
element.addEventListener("touchmove", dragOver);
});
element.addEventListener('touchend', () => {
dragging = false;
onDragEnd?.();
});
element.addEventListener('dragend', () => {
dragging = false;
onDragEnd?.();
});
element.addEventListener("click", () => {
if(currentlyZoomedInPost != null) {
zoomOutPost();
}
onClick?.();
});
}
function createText(parent, text, maxLines, randomSeed, fontIndex, maxWidth, textSizeFactor) {
const isPolaroid = polaroidTextMaxLines == maxLines;
maxWidth = maxWidth || textPostDefaultMaxWidth;
const widthFactor = maxWidth / textPostDefaultMaxWidth;
textSizeFactor = textSizeFactor || 1;
// max 32 bit int = 2,147,483,647
const prng = mulberry32(randomSeed);
const randNorm = () => (prng()-0.5)*2;
const fontFamily = [
"ZombieChecklistAlpha, Exposition",
"Exposition",
"DaisyRaeThin"
][fontIndex % 3];
const fontWeight = [
() => prng() > 0.5 ? "normal" : "bold",
() => prng() > 0.5 ? "normal" : "bold",
() => "bold"
][fontIndex % 3]();
const hue = prng();
let distanceFromTeal = ((hue - 0.25) * 2);
distanceFromTeal = distanceFromTeal < 0 ? -distanceFromTeal : distanceFromTeal;
distanceFromTeal = distanceFromTeal > 1 ? 1 : distanceFromTeal;
const color = hsvToRGBString(prng(), prng()*0.5+0.3, prng()*0.2+0.2 * (1-distanceFromTeal));
let textStyle = {fontFamily, fontWeight, fontSize: `${Math.round(textPostFontSizePixels*textSizeFactor)}px`};
let lineHeight = predictTextSize("TestjGg", textStyle)[1] * textPostLineSpacing;
const translateStart = textPostRandomTranslate.map(v => randNorm()*v);
const translateEnd = textPostRandomTranslate.map(v => randNorm()*v);
const bezierStart = textPostRandomBezier.map(v => randNorm()*v*widthFactor);
const bezierEnd = textPostRandomBezier.map(v => randNorm()*v*widthFactor);
const lineCurves = [...Array(maxLines)].map((_, i) => {
const lineY = (textPostPadding[1]*widthFactor*(isPolaroid?0:1))+((i+1)*lineHeight);
const lineTranslateStart = textLineRandomTranslate.map(v => randNorm()*v);
const lineTranslateEnd = textLineRandomTranslate.map(v => randNorm()*v);
const lineBezierStart = textLineRandomBezier.map(v => randNorm()*v*widthFactor);
const lineBezierEnd = textLineRandomBezier.map(v => randNorm()*v*widthFactor);
const start = [
textPostPadding[0]+translateStart[0]+lineTranslateStart[0],
lineY+((translateStart[1]+lineTranslateStart[1])*widthFactor)
];
const end = [
(maxWidth-textPostPadding[0])+translateEnd[0]+lineTranslateEnd[0],
lineY+((translateEnd[1]+lineTranslateEnd[1])*widthFactor)
];
const midpoint = [(start[0]+end[0])*0.5, (start[1]+end[1])*0.5];
return {
startPoint: `${start[0]} ${start[1]}`,
endPoint: `${end[0]*textPostTextLineLimitFudgeFactor} ${end[1]}`,
startHandle: `${midpoint[0]+bezierStart[0]+lineBezierStart[0]} ${midpoint[1]+(bezierStart[1]+lineBezierStart[1])}`,
endHandle: `${midpoint[0]+bezierEnd[0]+lineBezierEnd[0]} ${midpoint[1]+(bezierEnd[1]+lineBezierEnd[1])}`,
estimatedWidth: (end[0]-start[0]),
}
});
let widestLineWidth = 0;
let lines = text.split("\n").flatMap(line => {
const words = line.split(" ");
const linesToReturn = [];
let lineBuffer = [];
words.forEach(word => {
const textSize = predictTextSize([...lineBuffer, word].join(" "), textStyle);
const curveWidth = lineCurves[linesToReturn.length % lineCurves.length];
const addingCausesWrap = textSize[0]*textPostWidthEstimationFudgeFactor > curveWidth.estimatedWidth;
if(!addingCausesWrap && textSize[0] > widestLineWidth) {
widestLineWidth = textSize[0];
}
if(addingCausesWrap) {
widestLineWidth = maxWidth;
}
if(lineBuffer.length == 0 && addingCausesWrap) {
widestLineWidth = curveWidth.estimatedWidth;
linesToReturn.push([word]);
} else if(addingCausesWrap) {
linesToReturn.push(lineBuffer.join(" ")+"\n");
lineBuffer = [word];
} else {
lineBuffer.push(word);
}
});
if(lineBuffer.length && linesToReturn.length < maxLines) {
linesToReturn.push(lineBuffer.join(" "));
}
return linesToReturn;
});
let textPostWidth = (widestLineWidth*textPostWidthEstimationFudgeFactor)+textPostPadding[0]*2;
textPostWidth = textPostWidth > maxWidth ? maxWidth : textPostWidth;
const widthUtilizationFactor = textPostWidth/maxWidth;
const textPostHeight = ((lines.length > maxLines ? maxLines : lines.length) * lineHeight)+textPostPadding[1]*widthUtilizationFactor+textPostPadding[1];
const textPostVerticalPaddingClip = textPostPadding[1] - textPostPadding[1]*widthUtilizationFactor;
// if this text could be more than 30% wider without wrapping, start over with bigger font size.
if( textSizeFactor == 1 && widthUtilizationFactor < 0.7) {
textSizeFactor = 0.95/widthUtilizationFactor;
textSizeFactor = textSizeFactor > 2 ? 2 : textSizeFactor;
return createText(parent, text, maxLines, randomSeed, fontIndex, maxWidth, textSizeFactor)
}
// trim lines array to maxLines
if(lines.length > maxLines) {
lines = lines.slice(0, maxLines)
}
//lines[lines.length-1] += " ";
const textPreview = createElementNS(parent, svgXMLNS, "svg", {
"xmlns": [xmlnsXMLNS, svgXMLNS],
"xml:space": [xmlSpaceXMLNS, 'preserve'],
"version": "1.1",
"viewBox": `0 ${textPostVerticalPaddingClip} ${textPostWidth} ${textPostHeight}`,
"width": `${textPostWidth}`,
"height": `${textPostHeight}`,
});
lines.forEach((line, i) => {
const defsElement = createElementNS(textPreview, svgXMLNS, "defs");
const textElement = createElementNS(textPreview, svgXMLNS, "text");
createElementNS(
defsElement,
svgXMLNS,
"path",
{
id: `text-post-preview-path-${i}`,
d: `M ${lineCurves[i].startPoint} C ${lineCurves[i].startHandle}, ${lineCurves[i].endHandle}, ${lineCurves[i].endPoint}`,
}
)
createElementNS(
textElement,
svgXMLNS,
"textPath",
{
"xlink:href": [xmlXlinkXMLNS, `#text-post-preview-path-${i}`],
"style": {fontFamily, fontWeight, fontSize: `${Math.round(textPostFontSizePixels*textSizeFactor)}px`},
"fill": color
},
line,
)
});
}
function detectBPGSupport() {
if (window.localStorage['supportsBPG'] !== undefined) {
window.onBPGSupportDetected(Boolean(window.localStorage['supportsBPG']));
return
}
try {
setTimeout(() => window.onBPGSupportDetected(false), 1500);
const result = window.createBPGElement(document.body, testBPGImageDataURI, "display-none", 32, 32);
result.promise.then(
() => {
const context = result.canvas.getContext("2d");
const upperLeft = context.getImageData(0,0,1,1).data;
const upperRight = context.getImageData(31,0,1,1).data;
const lowerRight = context.getImageData(31,31,1,1).data;
const lowerLeft = context.getImageData(0,31,1,1).data;
const upperLeftIsRed = upperLeft[0] > 200 && upperLeft[1] < 50 && upperLeft[2] < 50;
const upperRightIsGreen = upperRight[0] < 50 && upperRight[1] > 200 && upperRight[2] < 50;
const lowerRightIsWhite = lowerRight[0] > 200 && lowerRight[1] > 200 && lowerRight[2] > 200;
const lowerLeftIsBlue = lowerLeft[0] < 50 && lowerLeft[1] < 50 && lowerLeft[2] > 200;
window.onBPGSupportDetected(upperLeftIsRed && upperRightIsGreen && lowerRightIsWhite && lowerLeftIsBlue);
},
() => {
window.onBPGSupportDetected(false);
}
)
} catch (err) {
window.onBPGSupportDetected(false);
}
}
function createElement(parent, tag, attr, textContent) {
const element = document.createElement(tag);
if(attr) {
Object.entries(attr).forEach(([k,v]) => {
if (v !== undefined) {
if(typeof v == 'function') {
element[k] = v;
} else if(typeof v == 'object') {
Object.entries(v).forEach(([k1,v1]) => {
element[k][k1] = v1;
});
} else {
element.setAttribute(k, v);
}
}
});
}
if(textContent) {
element.textContent = textContent;
}
parent.appendChild(element);
return element;
}
function createElementNS(parent, ns, tag, attr, textContent) {
const element = document.createElementNS(ns, tag);
if(attr) {
Object.entries(attr).forEach(([k,v]) => {
if(v !== undefined) {
if((typeof v) == "string") {
element.setAttributeNS(null, k, v)
} else if(typeof v == 'object' && !(v instanceof Array)) {
Object.entries(v).forEach(([k1,v1]) => {
element[k][k1] = v1;
});
} else {
const [ns, value] = v;
element.setAttributeNS(ns, k, value)
}
}
});
}
if(textContent) {
element.textContent = textContent;
}
parent.appendChild(element);
return element;
}
function appendFragment(parent, textContent) {
const fragment = document.createDocumentFragment()
fragment.textContent = textContent
parent.appendChild(fragment)
}
function predictTextSize(text, style) {
const measurementDiv = createElement(document.body, "div", {class: "predict-text-width"}, text)
Object.entries(style).forEach(([k,v]) => measurementDiv.style[k] = v);
const toReturn = [measurementDiv.clientWidth + 1, measurementDiv.clientHeight + 1];
document.body.removeChild(measurementDiv);
return toReturn;
}
function debounce(func, wait, immediate) {
var timeout;
return () => {
var context = this, args = arguments;
var later = () => {
timeout = null;
if (!immediate) func.apply(context, args);
};
var callNow = immediate && !timeout;
clearTimeout(timeout);
timeout = setTimeout(later, wait);
if (callNow) func.apply(context, args);
};
};
function xhr(method, url, body, callback) {
var request = new XMLHttpRequest();
request.addEventListener("load", function() {
callback(this.status, this.responseText);
});
request.open(method, url);
if(body && typeof body === 'object' && !(body instanceof FormData)) {
request.setRequestHeader("Content-Type", "application/json;charset=UTF-8");
body = JSON.stringify(body);
}
request.send(body);
}
// https://stackoverflow.com/questions/521295/seeding-the-random-number-generator-in-javascript
function mulberry32(a) {
return function() {
var t = a += 0x6D2B79F5;
t = Math.imul(t ^ t >>> 15, t | 1);
t ^= t + Math.imul(t ^ t >>> 7, t | 61);
return ((t ^ t >>> 14) >>> 0) / 4294967296;
}
}
//remove
function clearElementContents(element) {
while(element.lastChild) {
element.removeChild(element.lastChild);
}
}
function string2Hex(input) {
var str = '';
for(var i = 0; i < input.length; i++) {
str += input[i].charCodeAt(0).toString(16);
}
return str;
}
function hsvToRGBString(h, s, v) {
var r, g, b, i, f, p, q, t;
if (arguments.length === 1) {
s = h.s, v = h.v, h = h.h;
}
i = Math.floor(h * 6);
f = h * 6 - i;
p = v * (1 - s);
q = v * (1 - f * s);
t = v * (1 - (1 - f) * s);
switch (i % 6) {
case 0: r = v, g = t, b = p; break;
case 1: r = q, g = v, b = p; break;
case 2: r = p, g = v, b = t; break;
case 3: r = p, g = q, b = v; break;
case 4: r = t, g = p, b = v; break;
case 5: r = v, g = p, b = q; break;
}
return `rgb(${Math.round(r * 255)}, ${Math.round(g * 255)}, ${Math.round(b * 255)})`;
}
function clamp01(x) {
return x > 1 ? 1 : (x < 0 ? 0 : x);
}
function lerp(a,b, lerp) {
lerp = clamp01(lerp);
return a*(1-lerp)+b*lerp;
}
function intersectTwoLines(line1, line2) {
var d = (line1.vx * line2.vy) - (line1.vy * line2.vx);
if(d == 0) {
return [null, null];
}
var u1 = (line1.x1 * line1.y2 - line1.y1 * line1.x2);
var u2 = (line2.x1 * line2.y2 - line2.y1 * line2.x2);
var px = (u1 * line2.vx - line1.vx * u2) / d;
var py = (u1 * line2.vy - line1.vy * u2) / d;
return [ px, py ];
}
function dot(v1x, v1y, v2x, v2y) {
return v1x * v2x + v1y * v2y;
}
function magnitude(x, y) {
return Math.sqrt(sqrMagnitude(x, y));
}
function sqrMagnitude(x, y) {
return x * x + y * y;
}
function normalize(x, y) {
const m = magnitude(x, y);
return [x/m, y/m];
}
function getRandomPermutation (arr) {
const perm = Array.apply(null, {length: arr.length}).map((_,i) => [Math.random(), i]).sort().map(v => v[1]);
return arr.map((_, i) => arr[perm[i]]);
}
// https://stackoverflow.com/questions/7584794/accessing-jpeg-exif-rotation-data-in-javascript-on-the-client-side
// function getJPEGEXIFOrientationFromFile(file, callback) {
// var reader = new FileReader();
// reader.onload = function(e) {
// var view = new DataView(e.target.result);
// if (view.getUint16(0, false) != 0xFFD8)
// {
// return callback(-2);
// }
// var length = view.byteLength, offset = 2;
// while (offset < length)
// {
// if (view.getUint16(offset+2, false) <= 8) return callback(-1);
// var marker = view.getUint16(offset, false);
// offset += 2;
// if (marker == 0xFFE1)
// {
// if (view.getUint32(offset += 2, false) != 0x45786966)
// {
// return callback(-1);
// }
// var little = view.getUint16(offset += 6, false) == 0x4949;
// offset += view.getUint32(offset + 4, little);
// var tags = view.getUint16(offset, little);
// offset += 2;
// for (var i = 0; i < tags; i++)
// {
// if (view.getUint16(offset + (i * 12), little) == 0x0112)
// {
// return callback(view.getUint16(offset + (i * 12) + 8, little));
// }
// }
// }
// else if ((marker & 0xFF00) != 0xFF00)
// {
// break;
// }
// else
// {
// offset += view.getUint16(offset, false);
// }
// }
// return callback(-1);
// };
// reader.readAsArrayBuffer(file);
// }
})(window, document, window.graffitiAppState);