FEATURE script builder

This commit is contained in:
Pingumask 2024-08-05 21:46:18 +02:00
parent 3d9ec7c1ef
commit 16b77e95ea
6 changed files with 220 additions and 60 deletions

View file

@ -42,7 +42,7 @@
<script> <script>
import { mapState } from "vuex"; import { mapState } from "vuex";
import { version } from "../package.json"; import app from "../package.json";
import TownSquare from "./components/TownSquare"; import TownSquare from "./components/TownSquare";
import TownInfo from "./components/TownInfo"; import TownInfo from "./components/TownInfo";
import Menu from "./components/Menu"; import Menu from "./components/Menu";
@ -90,7 +90,7 @@ export default {
}, },
data() { data() {
return { return {
version, version: app.version,
}; };
}, },
methods: { methods: {

View file

@ -17,13 +17,6 @@
:class="{ active: tab == 'popular' }" :class="{ active: tab == 'popular' }"
>{{ locale.modal.edition.tab.popular }}</span >{{ locale.modal.edition.tab.popular }}</span
> >
<span
class="tab"
icon="theater-masks"
@click="tab = 'teensyville'"
:class="{ active: tab == 'teensyville' }"
>{{ locale.modal.edition.tab.teensyville }}</span
>
<span <span
class="tab" class="tab"
icon="question" icon="question"
@ -31,6 +24,16 @@
:class="{ active: tab == 'custom' }" :class="{ active: tab == 'custom' }"
>{{ locale.modal.edition.tab.custom }}</span >{{ locale.modal.edition.tab.custom }}</span
> >
<span
class="tab"
icon="question"
@click="
initPool();
tab = 'build';
"
:class="{ active: tab == 'build' }"
>{{ locale.modal.edition.tab.build }}</span
>
</li> </li>
<template v-if="tab == 'official'"> <template v-if="tab == 'official'">
<ul class="editions"> <ul class="editions">
@ -60,8 +63,7 @@
{{ script[0] }} {{ script[0] }}
</li> </li>
</ul> </ul>
</template> <h3>{{ locale.modal.edition.tab.teensyville }}</h3>
<template v-if="tab == 'teensyville'">
<ul class="scripts"> <ul class="scripts">
<li <li
v-for="(script, index) in editions.teensyville" v-for="(script, index) in editions.teensyville"
@ -110,6 +112,46 @@
</div> </div>
</div> </div>
</template> </template>
<template v-if="tab == 'build'">
<section
v-for="team in teams"
:key="team"
class="build team"
:class="team"
>
<aside class="aside">
<div>
<h4>{{ locale.modal.reference.teamNames[team] }}</h4>
<strong>{{ selectedInTeam(team) }}</strong>
</div>
</aside>
<ul class="roles" :class="team">
<li
v-for="role in rolesForTeam(team)"
class="role"
:class="{ selected: role.selected }"
:key="role.id"
@click="toggleRole(role.id)"
>
<Token :role="role" />
</li>
</ul>
</section>
<div class="button-group">
<div class="button" @click="resetBuilt">
<font-awesome-icon :icon="['fas', 'trash-alt']" />
</div>
<div class="button" @click="initPool">
<font-awesome-icon :icon="['fas', 'redo-alt']" />
</div>
<div class="button" @click="randomizeBuilt">
<font-awesome-icon :icon="['fas', 'random']" />
</div>
<div class="button" @click="startBuilt">
<font-awesome-icon :icon="['fas', 'play']" />
</div>
</div>
</template>
</ul> </ul>
</Modal> </Modal>
</template> </template>
@ -117,38 +159,95 @@
<script> <script>
import { mapMutations, mapState } from "vuex"; import { mapMutations, mapState } from "vuex";
import Modal from "./Modal"; import Modal from "./Modal";
import { rolesJSON } from "../../store/modules/locale";
import Token from "../Token";
export default { export default {
components: { components: {
Modal, Modal,
Token,
}, },
data: function () { data() {
return { return {
tab: "official", tab: "official",
draftPool: rolesJSON,
teams: ["townsfolk", "outsider", "minion", "demon"],
recommendedTeamSize: {
townsfolk: 13,
outsider: 4,
minion: 4,
demon: 4,
},
}; };
}, },
computed: { computed: {
...mapState(["modals", "locale", "editions", "roles", "jinxes"]), ...mapState(["modals", "locale", "editions", "roles", "jinxes"]),
}, },
methods: { methods: {
initPool() {
console.log("init pool");
console.table(this.roles);
this.draftPool = rolesJSON;
this.resetBuilt();
for (let [role] of this.roles) {
this.toggleRole(role);
}
},
toggleRole(id) {
const role = this.draftPool.find((r) => r.id === id);
if (role) {
this.$set(role, "selected", !role.selected);
}
},
rolesForTeam(team) {
return this.draftPool?.filter((role) => role.team === team) ?? [];
},
selectedInTeam(team) {
return this.draftPool?.filter(
(role) => role.team === team && role.selected,
).length;
},
resetBuilt() {
for (let role of this.draftPool) {
this.$set(role, "selected", false);
}
},
randomizeBuilt() {
this.resetBuilt();
for (let team of this.teams) {
let currentPool = this.rolesForTeam(team);
for (let i = 0; i < this.recommendedTeamSize[team]; i++) {
let picked = currentPool.splice(
Math.floor(Math.random() * currentPool.length),
1,
)[0];
this.$set(picked, "selected", true);
}
}
},
startBuilt() {
const selected = this.draftPool.filter((role) => role.selected);
this.parseRoles(selected);
},
openUpload() { openUpload() {
this.$refs.upload.click(); this.$refs.upload.click();
}, },
handleUpload() { handleUpload() {
const file = this.$refs.upload.files[0]; const file = this.$refs.upload.files[0];
if (file && file.size) { if (!file?.size) {
const reader = new FileReader(); return;
reader.addEventListener("load", () => {
try {
const roles = JSON.parse(reader.result);
this.parseRoles(roles);
} catch (e) {
alert("Error reading custom script: " + e.message);
}
this.$refs.upload.value = "";
});
reader.readAsText(file);
} }
const reader = new FileReader();
reader.addEventListener("load", () => {
try {
const uploadedRoles = JSON.parse(reader.result);
this.parseRoles(uploadedRoles);
} catch (e) {
alert(`Error reading custom script: ${e.message}`);
}
this.$refs.upload.value = "";
});
reader.readAsText(file);
}, },
promptURL() { promptURL() {
const url = prompt(this.locale.prompt.customUrl); const url = prompt(this.locale.prompt.customUrl);
@ -158,46 +257,40 @@ export default {
}, },
async handleURL(url) { async handleURL(url) {
const res = await fetch(url); const res = await fetch(url);
if (res && res.json) { if (res?.json) {
try { try {
const script = await res.json(); const script = await res.json();
this.parseRoles(script); this.parseRoles(script);
} catch (e) { } catch (e) {
alert(this.locale.prompt.customError + ": " + e.message); alert(`${this.locale.prompt.customError}: ${e.message}`);
} }
} }
}, },
async readFromClipboard() { async readFromClipboard() {
const text = await navigator.clipboard.readText(); const text = await navigator.clipboard.readText();
try { try {
const roles = JSON.parse(text); const uploadedRoles = JSON.parse(text);
this.parseRoles(roles); this.parseRoles(uploadedRoles);
} catch (e) { } catch (e) {
alert("Error reading custom script: " + e.message); alert(`Error reading custom script: ${e.message}`);
} }
}, },
parseRoles(roles) { parseRoles(pickedRoles) {
if (!roles || !roles.length) return; if (!pickedRoles || !pickedRoles.length) return;
roles = roles.map((role) => pickedRoles = pickedRoles.map((role) =>
typeof role === "string" ? { id: role } : role, typeof role === "string" ? { id: role } : role,
); );
const metaIndex = roles.findIndex(({ id }) => id === "_meta"); const metaIndex = pickedRoles.findIndex(({ id }) => id === "_meta");
let meta = {}; const meta = metaIndex > -1 ? pickedRoles.splice(metaIndex, 1).pop() : {};
if (metaIndex > -1) { this.$store.commit("setCustomRoles", pickedRoles);
meta = roles.splice(metaIndex, 1).pop(); this.$store.commit("setEdition", { ...meta, id: "custom" });
}
this.$store.commit("setCustomRoles", roles);
this.$store.commit(
"setEdition",
Object.assign({}, meta, { id: "custom" }),
);
// set fabled // set fabled
const fabled = []; const fabled = [];
var djinnAdded = false; let djinnAdded = false;
var djinnNeeded = false; let djinnNeeded = false;
var bootleggerAdded = false; let bootleggerAdded = false;
var bootleggerNedded = false; let bootleggerNedded = false;
roles.forEach((role) => { pickedRoles.forEach((role) => {
if (this.$store.state.fabled.has(role.id || role)) { if (this.$store.state.fabled.has(role.id || role)) {
fabled.push(this.$store.state.fabled.get(role.id || role)); fabled.push(this.$store.state.fabled.get(role.id || role));
if ((role.id || role) == "djinn") { if ((role.id || role) == "djinn") {
@ -240,6 +333,7 @@ export default {
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "../../vars.scss";
ul { ul {
width: 100%; width: 100%;
} }
@ -269,6 +363,14 @@ ul.editions {
} }
} }
.build .role {
width: 4vmax;
opacity: 0.7;
&.selected {
opacity: 1;
}
}
.tabs { .tabs {
display: flex; display: flex;
padding: 0; padding: 0;
@ -313,8 +415,6 @@ input[type="file"] {
list-style-type: disc; list-style-type: disc;
font-size: 120%; font-size: 120%;
cursor: pointer; cursor: pointer;
// display: grid;
// grid-template-columns: repeat(auto-fit, minmax(400px, 1fr));
display: flex; display: flex;
gap: 0.75em 1em; gap: 0.75em 1em;
justify-content: flex-start; justify-content: flex-start;
@ -331,4 +431,59 @@ input[type="file"] {
} }
} }
} }
.townsfolk {
aside {
background: linear-gradient(-90deg, $townsfolk, transparent);
}
}
.outsider {
aside {
background: linear-gradient(-90deg, $outsider, transparent);
}
}
.minion {
aside {
background: linear-gradient(-90deg, $minion, transparent);
}
}
.demon {
aside {
background: linear-gradient(-90deg, $demon, transparent);
}
}
.team {
display: grid;
width: 100%;
grid-template-columns: 3rem 1fr;
aside {
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
text-transform: uppercase;
font-size: 0.7rem;
strong {
display: block;
font-size: 1.4rem;
}
div {
font-size: 1.1rem;
text-align: center;
rotate: 90deg;
}
h4 {
margin-block: 0.25rem;
}
}
.roles {
display: flex;
flex-wrap: wrap;
justify-content: flex-start;
padding-left: 1rem;
padding-block: 0.25rem;
overflow: hidden;
}
}
</style> </style>

View file

@ -65,7 +65,6 @@ export default {
box-shadow: 2px 2px 20px 1px #000; box-shadow: 2px 2px 20px 1px #000;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
max-height: 95vh;
width: 90vw; width: 90vw;
max-width: 1800px; max-width: 1800px;
@ -73,7 +72,9 @@ export default {
.vote-history &, .vote-history &,
.night-reference &, .night-reference &,
.characters & { .characters & {
overflow-y: auto; overflow: auto;
max-height: 95vh;
scrollbar-gutter: stable both-edges;
} }
.roles &, .roles &,

View file

@ -34,6 +34,7 @@ const faIcons = [
"MinusSquare", "MinusSquare",
"Music", "Music",
"PeopleArrows", "PeopleArrows",
"Play",
"PlusCircle", "PlusCircle",
"Question", "Question",
"Random", "Random",

View file

@ -168,10 +168,11 @@
"edition": { "edition": {
"title": "Select an edition:", "title": "Select an edition:",
"tab": { "tab": {
"official": "Official scripts", "official": "Official",
"popular": "Popular scripts", "popular": "Popular",
"teensyville": "Teensyville", "teensyville": "Teensyville",
"custom": "Load custom" "custom": "Custom",
"build": "Build"
}, },
"custom": { "custom": {
"introStart": "To play with a custom script, you need to select the characters you want to play with in the official", "introStart": "To play with a custom script, you need to select the characters you want to play with in the official",
@ -277,4 +278,4 @@
"customMessages": ["","The debate is open","(Custom)"] "customMessages": ["","The debate is open","(Custom)"]
} }
} }
} }

View file

@ -168,10 +168,11 @@
"edition": { "edition": {
"title": "Choisir un Scénario :", "title": "Choisir un Scénario :",
"tab": { "tab": {
"official": "Scénarios officiels", "official": "Officiels",
"popular": "Scripts populaires", "popular": "Populaires",
"teensyville": "Teensyville", "teensyville": "Teensyville",
"custom": "Partie personnalisée" "custom": "Personnaliser",
"build": "Créer"
}, },
"custom": { "custom": {
"introStart": "Pour jouer avec un script personnalisé, vous pouvez sélectionner les personnages de votre choix grace à l'", "introStart": "Pour jouer avec un script personnalisé, vous pouvez sélectionner les personnages de votre choix grace à l'",
@ -185,7 +186,8 @@
"upload": "Téléversersement JSON", "upload": "Téléversersement JSON",
"url": "Entrer une URL", "url": "Entrer une URL",
"clipboard": "Presse-papier", "clipboard": "Presse-papier",
"back": "Retour" "back": "Retour",
"startDraft": "Démarrer un draft"
} }
}, },
"fabled": { "fabled": {
@ -277,4 +279,4 @@
"customMessages": ["","Le débat est ouvert","(Personnalisé)"] "customMessages": ["","Le débat est ouvert","(Personnalisé)"]
} }
} }
} }