Integrate backend with UI
This commit is contained in:
parent
400c11d3a7
commit
6efe880528
@ -3,8 +3,9 @@ import {initializeApp} from "firebase-admin/app";
|
|||||||
|
|
||||||
initializeApp();
|
initializeApp();
|
||||||
|
|
||||||
import votingRoute from "./route/voting";
|
import votingRoute, {activeVoteState} from "./route/voting";
|
||||||
import adminRoute from "./route/admin";
|
import adminRoute from "./route/admin";
|
||||||
|
|
||||||
export const vote = functions.https.onRequest(votingRoute);
|
export const vote = functions.https.onRequest(votingRoute);
|
||||||
export const admin = functions.https.onRequest(adminRoute);
|
export const admin = functions.https.onRequest(adminRoute);
|
||||||
|
export const activeVote = activeVoteState;
|
||||||
|
@ -9,6 +9,7 @@ app.use(cors());
|
|||||||
app.post("/create", async (req: Request, res: Response) => {
|
app.post("/create", async (req: Request, res: Response) => {
|
||||||
const prompt = req.body.prompt as string | undefined;
|
const prompt = req.body.prompt as string | undefined;
|
||||||
const options = req.body.options as Array<string> | undefined;
|
const options = req.body.options as Array<string> | undefined;
|
||||||
|
const description = req.body.description as string | undefined;
|
||||||
|
|
||||||
if (!prompt) {
|
if (!prompt) {
|
||||||
res.status(400).json({error: "Missing prompt"});
|
res.status(400).json({error: "Missing prompt"});
|
||||||
@ -20,7 +21,12 @@ app.post("/create", async (req: Request, res: Response) => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
res.json({id: await createVote(prompt, options)});
|
if (!description) {
|
||||||
|
res.status(400).json({error: "Missing description"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
res.json({id: await createVote(prompt, description, options)});
|
||||||
});
|
});
|
||||||
|
|
||||||
app.put("/setActive", async (req: Request, res: Response) => {
|
app.put("/setActive", async (req: Request, res: Response) => {
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import * as functions from "firebase-functions";
|
||||||
import express, {Request, Response} from "express";
|
import express, {Request, Response} from "express";
|
||||||
import {
|
import {
|
||||||
getEntriesFromSnapshot,
|
getEntriesFromSnapshot,
|
||||||
@ -122,4 +123,17 @@ app.get("/active", async (req: Request, res: Response) => {
|
|||||||
res.json({id: (await getActiveVote())?.id});
|
res.json({id: (await getActiveVote())?.id});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const activeVoteState = functions.https.onCall(async () => {
|
||||||
|
const vote = await getActiveVote();
|
||||||
|
if (!vote) return undefined;
|
||||||
|
|
||||||
|
const voteData = await vote.get();
|
||||||
|
return {
|
||||||
|
...voteData.data(),
|
||||||
|
voteId: vote.id,
|
||||||
|
votes: (await Promise.all(getVoteCounts(voteData)))
|
||||||
|
.map((query) => query.size),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
export default app;
|
export default app;
|
||||||
|
@ -20,6 +20,7 @@ export type VoteEntry = {
|
|||||||
|
|
||||||
export type Vote = {
|
export type Vote = {
|
||||||
prompt: string
|
prompt: string
|
||||||
|
description: string
|
||||||
[optionsField]: Array<string>
|
[optionsField]: Array<string>
|
||||||
[activeField]?: boolean
|
[activeField]?: boolean
|
||||||
};
|
};
|
||||||
@ -44,14 +45,17 @@ export const updateVoteEntry = async (
|
|||||||
vote: DocumentReference<VoteEntry>,
|
vote: DocumentReference<VoteEntry>,
|
||||||
voteIndex: number
|
voteIndex: number
|
||||||
) =>
|
) =>
|
||||||
vote.update({voteIndex});
|
vote.update({[voteIndexField]: voteIndex});
|
||||||
export const makeVoteEntry = async (
|
export const makeVoteEntry = async (
|
||||||
vote: DocumentReference<Vote>,
|
vote: DocumentReference<Vote>,
|
||||||
username: string,
|
username: string,
|
||||||
voteIndex: number
|
voteIndex: number
|
||||||
) =>
|
) =>
|
||||||
vote.collection(entriesCollectionName)
|
vote.collection(entriesCollectionName)
|
||||||
.add({username, voteIndex}) as Promise<DocumentReference<VoteEntry>>;
|
.add({
|
||||||
|
[usernameField]: username,
|
||||||
|
[voteIndexField]: voteIndex,
|
||||||
|
}) as Promise<DocumentReference<VoteEntry>>;
|
||||||
|
|
||||||
export const getVoteCounts = (
|
export const getVoteCounts = (
|
||||||
vote: DocumentSnapshot<Vote>
|
vote: DocumentSnapshot<Vote>
|
||||||
@ -80,8 +84,16 @@ export const getEntriesFromSnapshot = async (
|
|||||||
).docs.map((doc) => doc.data() as VoteEntry);
|
).docs.map((doc) => doc.data() as VoteEntry);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const createVote = async (prompt: string, options: Array<string>) => {
|
export const createVote = async (
|
||||||
const vote = await votesCollection.add({prompt, options});
|
prompt: string,
|
||||||
|
description: string,
|
||||||
|
options: Array<string>
|
||||||
|
) => {
|
||||||
|
const vote = await votesCollection.add({
|
||||||
|
prompt,
|
||||||
|
description,
|
||||||
|
[optionsField]: options,
|
||||||
|
});
|
||||||
return vote.id;
|
return vote.id;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -99,6 +99,12 @@
|
|||||||
<input type="text" class="form-row-field-input" placeholder=" " name="POST-prompt">
|
<input type="text" class="form-row-field-input" placeholder=" " name="POST-prompt">
|
||||||
<label for="POST-prompt" class="form-row-field">Prompt</label>
|
<label for="POST-prompt" class="form-row-field">Prompt</label>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="form-row">
|
||||||
|
<span class="form-row-number"></span>
|
||||||
|
|
||||||
|
<input type="text" class="form-row-field-input" placeholder=" " name="POST-description">
|
||||||
|
<label for="POST-prompt" class="form-row-field">Description</label>
|
||||||
|
</div>
|
||||||
<div class="form-row">
|
<div class="form-row">
|
||||||
<span class="form-row-number"></span>
|
<span class="form-row-number"></span>
|
||||||
|
|
||||||
@ -129,7 +135,7 @@
|
|||||||
|
|
||||||
<script>
|
<script>
|
||||||
const DEBUG = true;
|
const DEBUG = true;
|
||||||
const target = (DEBUG ? "http://127.0.0.1:5001/" : "https://us-central1-streamchair-psychology.cloudfunctions.net/") + "streamchair-psychology/us-central1/";
|
const target = DEBUG ? "http://127.0.0.1:5001/streamchair-psychology/us-central1/" : "https://us-central1-streamchair-psychology.cloudfunctions.net/";
|
||||||
|
|
||||||
document.addEventListener('DOMContentLoaded', function() {
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
const createVoteForm = document.querySelector('#createVoteForm');
|
const createVoteForm = document.querySelector('#createVoteForm');
|
||||||
@ -139,12 +145,18 @@
|
|||||||
|
|
||||||
const formData = new FormData(createVoteForm);
|
const formData = new FormData(createVoteForm);
|
||||||
const prompt = formData.get("POST-prompt");
|
const prompt = formData.get("POST-prompt");
|
||||||
|
const description = formData.get("POST-description");
|
||||||
|
|
||||||
if (!prompt) {
|
if (!prompt) {
|
||||||
alert("Please enter a prompt");
|
alert("Please enter a prompt");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!description) {
|
||||||
|
alert("Please enter a description");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const options = [];
|
const options = [];
|
||||||
for (let i = 1; i <= 4; i++) {
|
for (let i = 1; i <= 4; i++) {
|
||||||
const option = formData.get(`POST-opt${i}`);
|
const option = formData.get(`POST-opt${i}`);
|
||||||
@ -172,6 +184,7 @@
|
|||||||
req.setRequestHeader("Content-Type", "application/json;charset=UTF-8");
|
req.setRequestHeader("Content-Type", "application/json;charset=UTF-8");
|
||||||
req.send(JSON.stringify({
|
req.send(JSON.stringify({
|
||||||
prompt,
|
prompt,
|
||||||
|
description,
|
||||||
options
|
options
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
@ -123,6 +123,17 @@
|
|||||||
/* background: #123123; */
|
/* background: #123123; */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#novote {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
flex: 1;
|
||||||
|
width: 80%;
|
||||||
|
padding: 8px 16px;
|
||||||
|
margin: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
#voteprompt {
|
#voteprompt {
|
||||||
margin-bottom: 1px;
|
margin-bottom: 1px;
|
||||||
margin-top: 5px;
|
margin-top: 5px;
|
||||||
@ -132,8 +143,10 @@
|
|||||||
list-style: none;
|
list-style: none;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
|
align-items: center;
|
||||||
padding: 0px;
|
padding: 0px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
overflow-x: hidden;
|
||||||
/* background-color: #ffa100; */
|
/* background-color: #ffa100; */
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -235,6 +248,10 @@
|
|||||||
outline-style: solid;
|
outline-style: solid;
|
||||||
outline-width: 2px;
|
outline-width: 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.hidden {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
@ -265,8 +282,8 @@
|
|||||||
<div id="voteinfo" class="outlinebox">
|
<div id="voteinfo" class="outlinebox">
|
||||||
<h1 id="voteprompt">Is OP the asshole?</h1>
|
<h1 id="voteprompt">Is OP the asshole?</h1>
|
||||||
<ul id="votes" class="outlinebox">
|
<ul id="votes" class="outlinebox">
|
||||||
<li style="width: 60%; background-color: var(--vote1);">/Yes (60%)</li>
|
<!--<li style="width: 60%; background-color: var(--vote1);">/Yes (60%)</li>
|
||||||
<li style="width: 40%; background-color: var(--vote4);">/No (40%)</li>
|
<li style="width: 40%; background-color: var(--vote4);">/No (40%)</li>-->
|
||||||
</ul>
|
</ul>
|
||||||
<div id="voteextrainfo">
|
<div id="voteextrainfo">
|
||||||
<div id="votedetails">
|
<div id="votedetails">
|
||||||
@ -291,9 +308,14 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div id="novote" class="outlinebox hidden">
|
||||||
|
<h1>There is currently no vote going on!</h1>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
const voteUpdateInterval = 1000;
|
||||||
document.addEventListener('DOMContentLoaded', function() {
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
//const loadEl = document.querySelector('#load');
|
//const loadEl = document.querySelector('#load');
|
||||||
// // 🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥
|
// // 🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥
|
||||||
@ -311,6 +333,94 @@
|
|||||||
//
|
//
|
||||||
// // 🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥
|
// // 🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥
|
||||||
|
|
||||||
|
let activeVoteData = undefined;
|
||||||
|
function updateVoteUI(fullRefresh) {
|
||||||
|
if (activeVoteData === undefined) {
|
||||||
|
document.getElementById("voteinfo").classList.add("hidden");
|
||||||
|
document.getElementById("novote").classList.remove("hidden");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
document.getElementById("novote").classList.add("hidden");
|
||||||
|
document.getElementById("voteinfo").classList.remove("hidden");
|
||||||
|
|
||||||
|
if (fullRefresh) {
|
||||||
|
document.getElementById("voteprompt").innerText = activeVoteData.prompt;
|
||||||
|
document.getElementById("postdescription").innerText = activeVoteData.description;
|
||||||
|
}
|
||||||
|
|
||||||
|
let totalVotes = 0;
|
||||||
|
for (let i = 0; i < activeVoteData.votes.length; i++) {
|
||||||
|
totalVotes += activeVoteData.votes[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
const voteOpts = document.getElementById("voteopts");
|
||||||
|
const votes = document.getElementById("votes");
|
||||||
|
if (fullRefresh) {
|
||||||
|
// Remove all existing vote options
|
||||||
|
while (voteOpts.firstChild) {
|
||||||
|
voteOpts.removeChild(voteOpts.firstChild);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < activeVoteData.options.length; i++) {
|
||||||
|
const newOpt = document.createElement("li");
|
||||||
|
const newOptText = document.createElement("strong");
|
||||||
|
newOptText.innerText = activeVoteData.options[i];
|
||||||
|
newOpt.appendChild(newOptText);
|
||||||
|
voteOpts.appendChild(newOpt);
|
||||||
|
}
|
||||||
|
|
||||||
|
while (votes.firstChild) {
|
||||||
|
votes.removeChild(votes.firstChild);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < activeVoteData.options.length; i++) {
|
||||||
|
const newVote = document.createElement("li");
|
||||||
|
newVote.style.width = totalVotes == 0 ? `${100/activeVoteData.options.length}%` : `${(activeVoteData.votes[i] / totalVotes * 100)}%`;
|
||||||
|
newVote.style.backgroundColor = `var(--vote${i + 1})`;
|
||||||
|
newVote.innerHTML = activeVoteData.options[i] + " (" + (totalVotes == 0 ? `${+(100/activeVoteData.options.length).toFixed(2)}%` : +(activeVoteData.votes[i] / totalVotes * 100).toFixed(2)) + "%)";
|
||||||
|
votes.appendChild(newVote);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const voteOptsChildren = voteOpts.children;
|
||||||
|
for (let i = 0; i < voteOptsChildren.length; i++) {
|
||||||
|
const voteOpt = voteOptsChildren[i];
|
||||||
|
const voteOptText = voteOpt.children[0];
|
||||||
|
voteOptText.innerText = activeVoteData.options[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update votes list
|
||||||
|
const votesChildren = votes.children;
|
||||||
|
for (let i = 0; i < votesChildren.length; i++) {
|
||||||
|
const vote = votesChildren[i];
|
||||||
|
vote.innerHTML = activeVoteData.options[i] + " (" + (totalVotes == 0 ? `${+(100/activeVoteData.options.length).toFixed(2)}%` : +(activeVoteData.votes[i] / totalVotes * 100).toFixed(2)) + "%)";
|
||||||
|
vote.style.width = totalVotes == 0 ? `${100/activeVoteData.options.length}%` : (activeVoteData.votes[i] / totalVotes * 100) + "%";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.getElementById("optsandinfo").children[1].innerText = totalVotes + " votes registered";
|
||||||
|
}
|
||||||
|
|
||||||
|
const activeVote = firebase.functions().httpsCallable('activeVote');
|
||||||
|
function updateVote() {
|
||||||
|
activeVote().then((data) => {
|
||||||
|
if (data && data.data) {
|
||||||
|
if (!activeVoteData || activeVoteData.id !== data.id) {
|
||||||
|
activeVoteData = data.data;
|
||||||
|
updateVoteUI(true);
|
||||||
|
} else {
|
||||||
|
activeVoteData = data.data;
|
||||||
|
updateVoteUI(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setTimeout(updateVote, voteUpdateInterval);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
updateVoteUI(true);
|
||||||
|
updateVote();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
let app = firebase.app();
|
let app = firebase.app();
|
||||||
/*
|
/*
|
||||||
|
Loading…
x
Reference in New Issue
Block a user