singularity/main.js

575 lines
16 KiB
JavaScript
Raw Normal View History

2024-07-14 02:46:48 +02:00
var users = [];
var posts = {};
2024-08-04 16:02:19 +02:00
const adjectives = [
"dynamic", "strategic", "innovative", "passionate", "results-oriented",
"proactive", "visionary", "collaborative", "driven", "empathetic",
"adaptable", "resilient", "resourceful", "detail-oriented", "inspirational",
"analytical", "motivated", "solution-focused", "committed", "agile"
];
const interests = [
"music", "comedy", "travel", "technology", "hiking", "nature", "food", "movies",
"culture", "art", "activism", "community", "books", "baking", "creativity", "fitness",
"fashion", "wellness", "history", "adventure", "gaming", "gardening", "sustainability",
"coding", "coffee", "DIY", "crafts", "pets", "animals", "humor", "languages", "sports",
"competition", "meditation", "mindfulness", "design", "concerts", "innovation", "museums",
"future", "writing", "relaxation", "photography", "compassion", "nutrition", "style",
"restaurants", "gadgets", "inspiration", "literature", "outdoors", "wildlife", "conservation",
"motivation", "beauty", "culinary arts", "festivals", "exploration", "knowledge", "camping",
"archeology", "triathlon", "endurance", "luxury", "wine", "fine dining", "audio equipment",
"sound quality", "startups", "entrepreneurship", "extreme sports", "philosophy", "survival",
"environment", "politics", "running", "gastronomy", "genres", "software", "current events",
"recipes", "listening", "discovery", "expression", "reading", "recommendations", "news",
"analysis", "trends", "dining", "reviews", "rescue", "welfare", "health", "artists", "cooking"
];
var localUser = {
user: null,
interests: [],
postingStyle: null
};
var splashStep = 0;
const maxInterests = 3;
// configuration
2024-08-03 15:16:44 +02:00
const localMode = false;
const showSplash = false;
2024-07-17 01:08:03 +02:00
const blockContainer = document.getElementById("block-container");
2024-07-11 22:19:51 +02:00
const postCountElem = document.getElementById("post-count");
const postTotalElem = document.getElementById("post-total");
2024-07-11 21:13:17 +02:00
const loader = document.getElementById("loader");
// define a variable for how many posts we want to increase the page by
2024-07-11 22:19:51 +02:00
const postIncrease = 9;
2024-07-11 21:13:17 +02:00
// and define a value to determine which page we're on
let currentPage = 1;
2024-07-20 13:55:10 +02:00
// how many times can we increase the content until we reach the max limit?
function getPageCount() {
return Math.ceil(Object.keys(posts).length / postIncrease);
}
2024-07-11 21:13:17 +02:00
2024-07-11 22:19:51 +02:00
class Post {
// JSON post data
2024-07-14 02:46:48 +02:00
constructor(data) {
this.id = data.id;
this.username = data.associatedUser;
this.content = data.body;
this.replyTo = data.replyTo;
this.parentPost = null;
2024-07-14 02:46:48 +02:00
2024-08-04 16:18:12 +02:00
this.stars = 420;
this.replies = [];
2024-07-14 02:46:48 +02:00
}
getIsReply() {
return this.replyTo != "";
}
addReply(reply) {
this.replies.push(reply);
reply.parentPost = this;
2024-07-11 22:19:51 +02:00
}
getPostLevel() {
let p = this.parentPost;
let depth = 0;
while (p != null) {
p = p.parentPost;
depth++;
}
return depth;
}
2024-07-13 02:45:55 +02:00
getHeaderTag() {
2024-08-03 15:16:44 +02:00
return this.getPostLevel() == 0 ? "h1" : "h2";
}
2024-07-13 02:45:55 +02:00
getHeaderElement() {
2024-08-04 16:02:19 +02:00
const elem = document.createElement("div");
elem.className = "post-header";
2024-08-03 13:54:00 +02:00
// TODO: fetch current user pfp from thispersondoesnotexist and place in local storage
// for now if this person is us, post octopus
const currentUser = getCurrentUser();
const isCurrentUser = this.username == currentUser.user;
const pfpPath = isCurrentUser ? "oct.jpg" : `user/${this.username}.png`;
const pfpElem = document.createElement("img");
pfpElem.setAttribute("src", pfpPath);
pfpElem.setAttribute("class", "pfp");
elem.appendChild(pfpElem);
const usernameElem = document.createElement(this.getHeaderTag());
2024-08-03 15:16:44 +02:00
usernameElem.setAttribute("class", "username");
usernameElem.innerHTML = `<a href="#">${this.username}</a>`;
2024-08-03 13:54:00 +02:00
elem.appendChild(usernameElem);
2024-07-14 02:46:48 +02:00
return elem;
}
2024-07-14 02:46:48 +02:00
getContentElement() {
const elem = document.createElement("p");
elem.innerHTML = this.content;
2024-08-04 16:18:12 +02:00
return elem;
}
getIconElement(svg, right) {
const elem = document.createElement("div");
elem.className = "icon";
right *= 15;
elem.style.right = `${right}vw`;
2024-08-04 16:18:12 +02:00
const imgElem = document.createElement("img");
imgElem.className = "icon-img";
imgElem.setAttribute("src", svg);
elem.appendChild(imgElem);
const countElem = document.createElement("span");
// TODO: make an icon class to store count OR i guess just parse it out of the DOM
countElem.className = "icon-count";
countElem.innerHTML = "42069";
elem.appendChild(countElem);
return elem;
}
getFooterElement() {
const elem = document.createElement("div");
elem.className = "post-footer";
const starIconElem = this.getIconElement("icon/star-regular.svg", 0);
elem.appendChild(starIconElem);
const repostElem = this.getIconElement("icon/retweet-solid.svg", 1)
elem.appendChild(repostElem);
const bookmarkElem = this.getIconElement("icon/bookmark-regular.svg", 2);
elem.appendChild(bookmarkElem);
const commentElem = this.getIconElement("icon/comment-regular.svg", 3);
commentElem.addEventListener("click", () => writeReply(this));
elem.appendChild(commentElem);
2024-07-14 02:46:48 +02:00
2024-08-04 16:02:19 +02:00
return elem;
}
getElement() {
const elem = document.createElement("div");
2024-08-04 16:02:19 +02:00
elem.id = this.id;
2024-07-14 02:46:48 +02:00
// display root posts as blocks, and comments as attached to their posts
2024-08-03 15:16:44 +02:00
let classes = ["post"];
if (this.getPostLevel() == 0) {
2024-08-03 15:16:44 +02:00
classes.push("block");
}
2024-08-03 15:16:44 +02:00
elem.className = classes.join(" ");
elem.appendChild(this.getHeaderElement());
elem.appendChild(this.getContentElement());
2024-08-04 16:18:12 +02:00
elem.appendChild(this.getFooterElement());
2024-07-14 02:46:48 +02:00
for (let i = 0; i < this.replies.length; i++) {
const reply = this.replies[i];
elem.appendChild(reply.getElement());
2024-07-13 02:45:55 +02:00
}
return elem;
2024-07-11 22:19:51 +02:00
}
}
2024-07-14 02:46:48 +02:00
function getRootPosts() {
let result = [];
for (var id in posts) {
const post = posts[id];
if (post.getIsReply())
continue;
result.push(post);
}
return result;
2024-07-11 21:13:17 +02:00
}
2024-07-11 22:19:51 +02:00
function addPosts(pageIdx) {
2024-07-11 21:13:17 +02:00
currentPage = pageIdx;
2024-07-11 22:19:51 +02:00
const startRange = (pageIdx - 1) * postIncrease;
2024-07-20 13:55:10 +02:00
const endRange = currentPage == getPageCount()
? posts.length
2024-07-11 22:19:51 +02:00
: pageIdx * postIncrease;
2024-07-11 21:13:17 +02:00
2024-07-11 22:19:51 +02:00
postCountElem.innerHTML = endRange;
2024-07-11 21:13:17 +02:00
2024-07-14 02:46:48 +02:00
const rootPosts = getRootPosts();
2024-07-11 21:13:17 +02:00
for (let i = startRange + 1; i <= endRange; i++) {
const post = rootPosts[i];
const elem = post.getElement();
blockContainer.appendChild(elem);
2024-07-11 21:13:17 +02:00
}
}
function handleInfiniteScroll() {
throttle(() => {
const endOfPage = window.innerHeight + window.pageYOffset >= document.body.offsetHeight;
if (endOfPage) {
2024-07-11 22:19:51 +02:00
addPosts(currentPage + 1);
2024-07-11 21:13:17 +02:00
}
2024-07-20 13:55:10 +02:00
if (currentPage === getPageCount()) {
2024-07-11 21:13:17 +02:00
removeInfiniteScroll();
}
}, 1000);
}
2024-07-11 22:19:51 +02:00
// limit how often we try to load new posts to maintain browser performance
2024-07-11 21:13:17 +02:00
var throttleTimer;
function throttle(callback, time) {
if (throttleTimer) return;
throttleTimer = true;
setTimeout(() => {
callback();
throttleTimer = false;
}, time);
}
function removeInfiniteScroll() {
loader.remove();
window.removeEventListener("scroll", handleInfiniteScroll);
}
2024-07-20 14:40:54 +02:00
function getTopPost() {
// find the first post
const children = blockContainer.childNodes;
let firstPost = null;
for (let i = 0; i < children.length; i++) {
const child = children[i];
const classes = child.className.split(" ");
if (classes.some(c => c == "post")) {
firstPost = child;
break;
}
}
2024-07-18 02:16:52 +02:00
2024-07-20 14:40:54 +02:00
return firstPost;
}
function makePostFromJson(json) {
return new Post({
id: json.id,
associatedUser: json.associatedUser,
body: json.body
});
}
function getCurrentUser() {
// return some default values if we didn't do the initial configuration
if (!showSplash) {
return {
"user": "ktyl",
"interests": ["trains", "trains", "trains"],
"posting_style": "borderline maniacal train content"
};
}
2024-07-20 14:40:54 +02:00
return {
"user": localUser.user,
"interests": localUser.interests,
2024-07-20 14:40:54 +02:00
"posting_style": "just the most truly inane takes"
};
}
2024-07-18 02:16:52 +02:00
2024-08-04 20:33:00 +02:00
function getPostRequest(body) {
return {
2024-07-18 02:16:52 +02:00
method: "POST",
mode: "cors",
2024-08-04 20:33:00 +02:00
body: JSON.stringify(body),
2024-07-18 02:16:52 +02:00
headers: {
"Content-type": "application/json; charset=UTF-8"
}
2024-07-20 14:40:54 +02:00
};
2024-08-04 20:33:00 +02:00
}
2024-07-18 02:16:52 +02:00
2024-08-04 20:33:00 +02:00
function writePost() {
2024-08-03 13:28:23 +02:00
if (localMode) {
const post = new Post({
id: "1234",
associatedUser: getCurrentUser().user,
body: "local mode post (local mode post)"
});
2024-08-04 20:33:00 +02:00
blockContainer.insertBefore(post.getElement(), getTopPost());
return;
}
fetch("https://api.wayfarer.games/singularity/generate-posts.php", getPostRequest(getCurrentUser()))
.then(response => response.json())
.then(makePostFromJson)
.then(post => blockContainer.insertBefore(post.getElement(), getTopPost()));
2024-08-04 16:02:19 +02:00
}
function writeReply(post) {
// find the correct element
const elem = document.getElementById(post.id);
2024-08-04 20:33:00 +02:00
const user = getCurrentUser();
2024-08-04 16:02:19 +02:00
2024-08-04 20:33:00 +02:00
if (localMode) {
console.error("TODO: implement local replies");
return;
}
const replyBody = {
postId: post.id,
interests: user.interests,
user: user.user,
posting_style: user.posting_style
};
const request = getPostRequest(replyBody);
fetch("https://api.wayfarer.games/singularity/generate-reply.php", getPostRequest(replyBody))
.then(response => response.json())
.then(makePostFromJson)
.then(reply => {
post.addReply(reply);
elem.append(reply.getElement());
});
2024-08-04 16:02:19 +02:00
}
2024-07-17 01:08:55 +02:00
function addWritePostBlock() {
const blockElem = document.createElement("div");
2024-08-04 20:33:00 +02:00
blockElem.addEventListener("click", writePost);
2024-08-03 15:16:44 +02:00
blockElem.className = "block write-post";
2024-07-17 01:08:55 +02:00
2024-08-03 15:16:44 +02:00
const spanElem = document.createElement("h2");
spanElem.className = "";
spanElem.innerHTML = "Write something interesting for me!";
blockElem.append(spanElem);
2024-07-17 01:08:55 +02:00
2024-08-03 15:16:44 +02:00
blockContainer.append(blockElem);
2024-07-17 01:08:55 +02:00
}
2024-07-14 02:46:48 +02:00
function init() {
if (posts == undefined)
{
console.log("resource loading failed");
return;
}
// need to load all the resources first
2024-07-17 01:08:55 +02:00
const postCount = Object.keys(posts).length;
if (users.length == 0 || postCount == 0)
2024-07-14 02:46:48 +02:00
return;
2024-07-17 01:08:55 +02:00
console.log(`loaded ${users.length} users and ${postCount} posts`);
// TODO: add user bio above write post button
2024-07-17 01:08:55 +02:00
addWritePostBlock();
2024-07-14 02:46:48 +02:00
addPosts(currentPage);
window.addEventListener("scroll", handleInfiniteScroll);
}
function chooseInterest(interest) {
if (localUser.interests.length == maxInterests) {
console.error(`can't choose more than ${maxInterests} interests`);
return;
}
localUser.interests.push(interest);
const interestsTextElem = document.getElementById("interests-text");
if (localUser.interests.length != maxInterests) {
interestsTextElem.innerHTML = getInterestsTextValue(localUser.interests.length);
return;
}
const advanceButtonElem = document.getElementById("advance-button");
advanceButtonElem.innerHTML = "Begin!";
advanceButtonElem.style.visibility = "visible";
interestsTextElem.remove();
const interestsListElem = document.getElementById("interest-selection");
interestsListElem.remove();
}
function getInterestsSubset() {
const count = 20;
let subset = [];
while (subset.length < count) {
const interest = interests[Math.floor(Math.random() * interests.length)];
// skip if it's already included
if (subset.includes(interest))
continue;
// skip if the user has already chosen it
if (localUser.interests.includes(interest))
continue;
subset.push(interest);
}
return subset;
}
function populateSplashInterests() {
const rootElem = document.getElementById("interest-selection");
if (rootElem == null)
return;
// clear existing interests
rootElem.innerHTML = "";
const interestsSubset = getInterestsSubset();
for (let i = 0; i < interestsSubset.length; i++) {
const interest = interestsSubset[i];
const listItemElem = document.createElement("li");
rootElem.appendChild(listItemElem);
const buttonElem = document.createElement("a");
buttonElem.innerHTML = `#${interest}`;
buttonElem.addEventListener("click", () => {
chooseInterest(interest);
populateSplashInterests();
});
listItemElem.appendChild(buttonElem);
}
}
function getInterestsTextValue(numChosenInterests) {
return `Choose some interests! (${numChosenInterests}/${maxInterests})`;
}
function usernameInputUpdated(event) {
const inputElem = document.getElementById("username");
const buttonElem = document.getElementById("advance-button");
const isNameEmpty = inputElem.value.length == "";
buttonElem.style.visibility = isNameEmpty ? "hidden" : "visible";
if (isNameEmpty)
return;
if (event.key == "Enter") {
advanceSplash();
}
}
function chooseName() {
// check that a name has been chosen
const inputElem = document.getElementById("username");
if (!inputElem.value) {
console.error("a name needs to be entered!");
return;
}
splashStep = 1;
localUser.user = inputElem.value;
// TODO: disable button until name has been chosen
// TODO: insert interest selection elements before name input
//writePost(post => blockContainer.insertBefore(post.getElement(), getTopPost()));
const splashElem = document.getElementById("start-splash");
const interestsTextElem = document.createElement("p");
interestsTextElem.innerHTML = getInterestsTextValue(0);
interestsTextElem.id = "interests-text";
interestsTextElem.className = "center";
splashElem.insertBefore(interestsTextElem, inputElem);
const interestsListElem = document.createElement("ul");
interestsListElem.className = "center";
interestsListElem.id = "interest-selection";
splashElem.insertBefore(interestsListElem, inputElem);
populateSplashInterests();
// remove name input
inputElem.remove();
const advanceButtonElem = document.getElementById("advance-button");
advanceButtonElem.style.visibility = "hidden";
splashStep = 1;
}
function chooseInterests() {
if (localUser.interests.length < maxInterests) {
console.error(`need to choose ${maxInterests} interests`);
return;
}
console.log("TODO: generate user posting style");
}
function removeSplash() {
document.getElementById("start-splash").remove();
}
function advanceSplash() {
switch(splashStep) {
case 0:
chooseName();
break;
case 1:
chooseInterests();
removeSplash();
break;
default:
console.error(`nothing defined for splash step ${splashStep}`);
}
}
2024-07-14 02:46:48 +02:00
function loadDataFromEndpoint(endpoint, callback) {
fetch(endpoint)
.then(response => response.json())
.then(json => {
callback(json);
init();
});
}
2024-07-11 21:13:17 +02:00
2024-08-03 13:28:23 +02:00
const usersUrl = localMode ? "users.json" : "https://api.wayfarer.games/singularity/users.json";
const postsUrl = localMode ? "posts.json" : "https://api.wayfarer.games/singularity/posts.json";
loadDataFromEndpoint(usersUrl, json => { users = json.users; });
loadDataFromEndpoint(postsUrl, json => {
2024-07-14 02:46:48 +02:00
// first pass to instantiate all the posts
for (let i = 0; i < json.content.length; i++) {
const post = new Post(json.content[i]);
posts[post.id] = post;
}
// second pass to link each reply to the appropriate parent
for (const id in posts) {
const post = posts[id];
if (!post.getIsReply())
continue;
const parent = posts[post.replyTo];
parent.addReply(post);
}
2024-07-20 13:55:10 +02:00
postTotalElem.innerHTML = Object.keys(posts).length;
2024-07-14 02:46:48 +02:00
});
if (!showSplash) {
removeSplash();
}