Upgrade node to 22, bundle with vite
Updated docker compose for --watch usage
8
.github/workflows/linter.yml
vendored
|
@ -2,9 +2,9 @@ name: Lint Code Base
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches: [ main, develop ]
|
branches: [main, develop]
|
||||||
pull_request:
|
pull_request:
|
||||||
branches: [ main, develop ]
|
branches: [main, develop]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
|
@ -14,7 +14,7 @@ jobs:
|
||||||
- name: Setup node version
|
- name: Setup node version
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: '18'
|
node-version: "18"
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- run: npm install
|
- run: npm install
|
||||||
- run: npm run lint-ci
|
- run: npm run lint-check
|
||||||
|
|
|
@ -4,6 +4,12 @@
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Version 3.23.0
|
||||||
|
|
||||||
|
- Upgraded from node 18 to node 22
|
||||||
|
- Replaced vue-cli with Vite
|
||||||
|
- Set up for docker watch
|
||||||
|
|
||||||
### Version 3.22.0
|
### Version 3.22.0
|
||||||
|
|
||||||
- Official abilities rebalances :
|
- Official abilities rebalances :
|
||||||
|
|
10
Dockerfile
|
@ -1,11 +1,9 @@
|
||||||
FROM node:18
|
FROM node:22
|
||||||
RUN apt update && apt install -y\
|
RUN apt update && apt install -y\
|
||||||
git\
|
git\
|
||||||
&& apt clean
|
&& apt clean
|
||||||
WORKDIR /app/townsquare
|
WORKDIR /app/townsquare
|
||||||
COPY package*.json .
|
|
||||||
RUN npm rebuild
|
|
||||||
RUN npm clean-install
|
|
||||||
# npm rebuild avoids having misconfigurations if npm install has been run in the folder from windows before building the docker image
|
|
||||||
COPY . .
|
COPY . .
|
||||||
CMD ["npm","run","serve"]
|
RUN npm rebuild && npm clean-install
|
||||||
|
EXPOSE 5173 8079
|
||||||
|
CMD ["npm","run","dev"]
|
||||||
|
|
|
@ -2,10 +2,16 @@ module.exports = {
|
||||||
root: true,
|
root: true,
|
||||||
env: {
|
env: {
|
||||||
node: true,
|
node: true,
|
||||||
|
es2022: true,
|
||||||
|
browser: true,
|
||||||
},
|
},
|
||||||
extends: ["plugin:vue/essential", "eslint:recommended", "@vue/prettier"],
|
extends: [
|
||||||
|
"plugin:vue/essential",
|
||||||
|
"eslint:recommended",
|
||||||
|
"@vue/prettier"
|
||||||
|
],
|
||||||
parserOptions: {
|
parserOptions: {
|
||||||
ecmaVersion: 2020,
|
ecmaVersion: 2022,
|
||||||
},
|
},
|
||||||
rules: {
|
rules: {
|
||||||
"no-console": process.env.NODE_ENV === "production" ? 1 : 0,
|
"no-console": process.env.NODE_ENV === "production" ? 1 : 0,
|
|
@ -8,22 +8,22 @@
|
||||||
<meta name="mobile-web-app-capable" content="yes">
|
<meta name="mobile-web-app-capable" content="yes">
|
||||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
<title>Blood on the Clocktower Town Square</title>
|
<title>Blood on the Clocktower Town Square</title>
|
||||||
<link rel="apple-touch-icon" sizes="57x57" href="static/apple-icon-57x57.png">
|
<link rel="apple-touch-icon" sizes="57x57" href="/apple-icon-57x57.png">
|
||||||
<link rel="apple-touch-icon" sizes="60x60" href="static/apple-icon-60x60.png">
|
<link rel="apple-touch-icon" sizes="60x60" href="/apple-icon-60x60.png">
|
||||||
<link rel="apple-touch-icon" sizes="72x72" href="static/apple-icon-72x72.png">
|
<link rel="apple-touch-icon" sizes="72x72" href="/apple-icon-72x72.png">
|
||||||
<link rel="apple-touch-icon" sizes="76x76" href="static/apple-icon-76x76.png">
|
<link rel="apple-touch-icon" sizes="76x76" href="/apple-icon-76x76.png">
|
||||||
<link rel="apple-touch-icon" sizes="114x114" href="static/apple-icon-114x114.png">
|
<link rel="apple-touch-icon" sizes="114x114" href="/apple-icon-114x114.png">
|
||||||
<link rel="apple-touch-icon" sizes="120x120" href="static/apple-icon-120x120.png">
|
<link rel="apple-touch-icon" sizes="120x120" href="/apple-icon-120x120.png">
|
||||||
<link rel="apple-touch-icon" sizes="144x144" href="static/apple-icon-144x144.png">
|
<link rel="apple-touch-icon" sizes="144x144" href="/apple-icon-144x144.png">
|
||||||
<link rel="apple-touch-icon" sizes="152x152" href="static/apple-icon-152x152.png">
|
<link rel="apple-touch-icon" sizes="152x152" href="/apple-icon-152x152.png">
|
||||||
<link rel="apple-touch-icon" sizes="180x180" href="static/apple-icon-180x180.png">
|
<link rel="apple-touch-icon" sizes="180x180" href="/apple-icon-180x180.png">
|
||||||
<link rel="icon" type="image/png" sizes="192x192" href="static/android-icon-192x192.png">
|
<link rel="icon" type="image/png" sizes="192x192" href="/android-icon-192x192.png">
|
||||||
<link rel="icon" type="image/png" sizes="32x32" href="static/favicon-32x32.png">
|
<link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png">
|
||||||
<link rel="icon" type="image/png" sizes="96x96" href="static/favicon-96x96.png">
|
<link rel="icon" type="image/png" sizes="96x96" href="/favicon-96x96.png">
|
||||||
<link rel="icon" type="image/png" sizes="16x16" href="static/favicon-16x16.png">
|
<link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png">
|
||||||
<link rel="manifest" href="static/manifest.json">
|
<link rel="manifest" href="/manifest.json">
|
||||||
<meta name="msapplication-TileColor" content="#000000">
|
<meta name="msapplication-TileColor" content="#000000">
|
||||||
<meta name="msapplication-TileImage" content="/static/ms-icon-144x144.png">
|
<meta name="msapplication-TileImage" content="/ms-icon-144x144.png">
|
||||||
<meta name="theme-color" content="#ff0000">
|
<meta name="theme-color" content="#ff0000">
|
||||||
<link href="https://fonts.googleapis.com/css?family=Roboto+Condensed&display=swap" rel="stylesheet">
|
<link href="https://fonts.googleapis.com/css?family=Roboto+Condensed&display=swap" rel="stylesheet">
|
||||||
<meta name="description" content="A free, virtual Blood on the Clocktower Town Square and Grimoire to help you run (online) games both as a storyteller and player. Batteries included!">
|
<meta name="description" content="A free, virtual Blood on the Clocktower Town Square and Grimoire to help you run (online) games both as a storyteller and player. Batteries included!">
|
||||||
|
@ -41,5 +41,6 @@
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="app"></div>
|
<div id="app"></div>
|
||||||
|
<script type="module" src="./src/main.js"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
8557
package-lock.json
generated
40
package.json
|
@ -1,36 +1,36 @@
|
||||||
{
|
{
|
||||||
"name": "townsquare",
|
"name": "townsquare",
|
||||||
"version": "3.22.0",
|
"version": "3.23.0",
|
||||||
"description": "Blood on the Clocktower Town Square",
|
"description": "Blood on the Clocktower Town Square",
|
||||||
"author": "Pingumaskt",
|
"author": "Pingumaskt",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"serve": "vue-cli-service serve",
|
"dev": "vite --host 0.0.0.0",
|
||||||
"build": "vue-cli-service build ./src/main.js",
|
"build": "vite build",
|
||||||
"lint": "vue-cli-service lint",
|
"serve": "vite preview",
|
||||||
"lint-ci": "vue-cli-service lint --no-fix --max-warnings=0"
|
"lint": "eslint src --fix",
|
||||||
|
"lint-check": "eslint src --no-fix --max-warnings=0"
|
||||||
},
|
},
|
||||||
|
"type": "module",
|
||||||
"main": "App.vue",
|
"main": "App.vue",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fortawesome/fontawesome-svg-core": "^1.2.32",
|
"@fortawesome/fontawesome-svg-core": "^6.7.1",
|
||||||
"@fortawesome/free-brands-svg-icons": "^5.15.1",
|
"@fortawesome/free-brands-svg-icons": "^6.7.1",
|
||||||
"@fortawesome/free-solid-svg-icons": "^5.15.1",
|
"@fortawesome/free-solid-svg-icons": "^6.7.1",
|
||||||
"@fortawesome/vue-fontawesome": "^0.1.10",
|
"@fortawesome/vue-fontawesome": "^2.0.10",
|
||||||
"@vue/cli-service": "^5.0.8",
|
"@vitejs/plugin-vue2": "^2.3.3",
|
||||||
"prom-client": "^13.0.0",
|
"vite": "^6.0.3",
|
||||||
"vue": "^2.7.16",
|
"vue": "^2.7.16",
|
||||||
"vue-template-compiler": "^2.7.15",
|
"vue-template-compiler": "^2.7.15",
|
||||||
"vuex": "^3.6.2",
|
"vuex": "^3.6.2",
|
||||||
"ws": "^7.4.6"
|
"ws": "^8.18.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@vue/cli-plugin-eslint": "^5.0.8",
|
"@vue/eslint-config-prettier": "^10.1.0",
|
||||||
"@vue/eslint-config-prettier": "^8.0.0",
|
"eslint": "^9.16.0",
|
||||||
"eslint": "^8.53.0",
|
"eslint-plugin-prettier": "^5.2.1",
|
||||||
"eslint-plugin-prettier": "^5.0.1",
|
"eslint-plugin-vue": "^9.32.0",
|
||||||
"eslint-plugin-vue": "^9.18.1",
|
|
||||||
"prettier": "^3.0.3",
|
"prettier": "^3.0.3",
|
||||||
"sass": "^1.82.0",
|
"sass": "^1.82.0"
|
||||||
"sass-loader": "^16.0.4"
|
|
||||||
},
|
},
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"botc",
|
"botc",
|
||||||
|
@ -44,6 +44,6 @@
|
||||||
"url": "https://github.com//bra1n/townsquare.git"
|
"url": "https://github.com//bra1n/townsquare.git"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "^18"
|
"node": "^22"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Before Width: | Height: | Size: 32 KiB After Width: | Height: | Size: 32 KiB |
Before Width: | Height: | Size: 54 KiB After Width: | Height: | Size: 54 KiB |
Before Width: | Height: | Size: 3.8 KiB After Width: | Height: | Size: 3.8 KiB |
Before Width: | Height: | Size: 5.8 KiB After Width: | Height: | Size: 5.8 KiB |
Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 10 KiB |
Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 16 KiB |
Before Width: | Height: | Size: 22 KiB After Width: | Height: | Size: 22 KiB |
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 24 KiB |
Before Width: | Height: | Size: 32 KiB After Width: | Height: | Size: 32 KiB |
Before Width: | Height: | Size: 35 KiB After Width: | Height: | Size: 35 KiB |
Before Width: | Height: | Size: 47 KiB After Width: | Height: | Size: 47 KiB |
Before Width: | Height: | Size: 7.4 KiB After Width: | Height: | Size: 7.4 KiB |
Before Width: | Height: | Size: 7.9 KiB After Width: | Height: | Size: 7.9 KiB |
Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 10 KiB |
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 11 KiB |
Before Width: | Height: | Size: 55 KiB After Width: | Height: | Size: 55 KiB |
Before Width: | Height: | Size: 55 KiB After Width: | Height: | Size: 55 KiB |
Before Width: | Height: | Size: 1.7 KiB After Width: | Height: | Size: 1.7 KiB |
Before Width: | Height: | Size: 3.3 KiB After Width: | Height: | Size: 3.3 KiB |
Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 16 KiB |
Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 1.1 KiB |
Before Width: | Height: | Size: 32 KiB After Width: | Height: | Size: 32 KiB |
Before Width: | Height: | Size: 34 KiB After Width: | Height: | Size: 34 KiB |
Before Width: | Height: | Size: 108 KiB After Width: | Height: | Size: 108 KiB |
Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 10 KiB |
28
src/App.vue
|
@ -43,20 +43,20 @@
|
||||||
<script>
|
<script>
|
||||||
import { mapState } from "vuex";
|
import { mapState } from "vuex";
|
||||||
import app from "../package.json";
|
import app from "../package.json";
|
||||||
import TownSquare from "./components/TownSquare";
|
import TownSquare from "./components/TownSquare.vue";
|
||||||
import TownInfo from "./components/TownInfo";
|
import TownInfo from "./components/TownInfo.vue";
|
||||||
import Menu from "./components/Menu";
|
import Menu from "./components/Menu.vue";
|
||||||
import RolesModal from "./components/modals/RolesModal";
|
import RolesModal from "./components/modals/RolesModal.vue";
|
||||||
import EditionModal from "./components/modals/EditionModal";
|
import EditionModal from "./components/modals/EditionModal.vue";
|
||||||
import Intro from "./components/Intro";
|
import Intro from "./components/Intro.vue";
|
||||||
import ReferenceModal from "./components/modals/ReferenceModal";
|
import ReferenceModal from "./components/modals/ReferenceModal.vue";
|
||||||
import Vote from "./components/Vote";
|
import Vote from "./components/Vote.vue";
|
||||||
import Gradients from "./components/Gradients";
|
import Gradients from "./components/Gradients.vue";
|
||||||
import NightOrderModal from "./components/modals/NightOrderModal";
|
import NightOrderModal from "./components/modals/NightOrderModal.vue";
|
||||||
import FabledModal from "@/components/modals/FabledModal";
|
import FabledModal from "./components/modals/FabledModal.vue";
|
||||||
import VoteHistoryModal from "@/components/modals/VoteHistoryModal";
|
import VoteHistoryModal from "./components/modals/VoteHistoryModal.vue";
|
||||||
import GameStateModal from "@/components/modals/GameStateModal";
|
import GameStateModal from "./components/modals/GameStateModal.vue";
|
||||||
import SpecialVoteModal from "@/components/modals/SpecialVoteModal";
|
import SpecialVoteModal from "./components/modals/SpecialVoteModal.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: {
|
components: {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="intro">
|
<div class="intro">
|
||||||
<img src="static/apple-icon.png" alt="" class="logo" />
|
<img src="../../public/apple-icon.png" alt="" class="logo" />
|
||||||
<div>
|
<div>
|
||||||
{{ locale.intro.header }}
|
{{ locale.intro.header }}
|
||||||
<span class="button" @click="toggleMenu">
|
<span class="button" @click="toggleMenu">
|
||||||
|
|
|
@ -242,11 +242,7 @@
|
||||||
backgroundImage: `url(${
|
backgroundImage: `url(${
|
||||||
reminder.image && grimoire.isImageOptIn
|
reminder.image && grimoire.isImageOptIn
|
||||||
? reminder.image
|
? reminder.image
|
||||||
: require(
|
: rolePath(reminder.role)
|
||||||
'../assets/icons/' +
|
|
||||||
(reminder.imageAlt || reminder.role) +
|
|
||||||
'.png',
|
|
||||||
)
|
|
||||||
})`,
|
})`,
|
||||||
}"
|
}"
|
||||||
></span>
|
></span>
|
||||||
|
@ -261,7 +257,7 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import Token from "./Token";
|
import Token from "./Token.vue";
|
||||||
import { mapGetters, mapState } from "vuex";
|
import { mapGetters, mapState } from "vuex";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
@ -356,6 +352,12 @@ export default {
|
||||||
reminders.splice(this.player.reminders.indexOf(reminder), 1);
|
reminders.splice(this.player.reminders.indexOf(reminder), 1);
|
||||||
this.updatePlayer("reminders", reminders, true);
|
this.updatePlayer("reminders", reminders, true);
|
||||||
},
|
},
|
||||||
|
rolePath(role) {
|
||||||
|
return new URL(
|
||||||
|
`../assets/icons/${role.imageAlt || role.id}.png`,
|
||||||
|
import.meta.url,
|
||||||
|
).href;
|
||||||
|
},
|
||||||
updatePlayer(property, value, closeMenu = false) {
|
updatePlayer(property, value, closeMenu = false) {
|
||||||
if (
|
if (
|
||||||
this.session.isSpectator &&
|
this.session.isSpectator &&
|
||||||
|
|
|
@ -5,9 +5,7 @@
|
||||||
v-if="role.id"
|
v-if="role.id"
|
||||||
:style="{
|
:style="{
|
||||||
backgroundImage: `url(${
|
backgroundImage: `url(${
|
||||||
role.image && grimoire.isImageOptIn
|
role.image && grimoire.isImageOptIn ? role.image : rolePath
|
||||||
? role.image
|
|
||||||
: require('../assets/icons/' + (role.imageAlt || role.id) + '.png')
|
|
||||||
})`,
|
})`,
|
||||||
}"
|
}"
|
||||||
></span>
|
></span>
|
||||||
|
@ -64,6 +62,12 @@ export default {
|
||||||
(this.role.remindersGlobal || []).length
|
(this.role.remindersGlobal || []).length
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
rolePath() {
|
||||||
|
return new URL(
|
||||||
|
`../assets/icons/${this.role.imageAlt || this.role.id}.png`,
|
||||||
|
import.meta.url,
|
||||||
|
).href;
|
||||||
|
},
|
||||||
...mapState(["grimoire"]),
|
...mapState(["grimoire"]),
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
|
|
|
@ -5,9 +5,7 @@
|
||||||
:class="['edition-' + edition.id]"
|
:class="['edition-' + edition.id]"
|
||||||
:style="{
|
:style="{
|
||||||
backgroundImage: `url(${
|
backgroundImage: `url(${
|
||||||
edition.logo && grimoire.isImageOptIn
|
edition.logo && grimoire.isImageOptIn ? edition.logo : logoPath
|
||||||
? edition.logo
|
|
||||||
: require('../assets/logos/' + edition.id + '.png')
|
|
||||||
})`,
|
})`,
|
||||||
}"
|
}"
|
||||||
></li>
|
></li>
|
||||||
|
@ -102,15 +100,16 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import gameJSON from "./../game";
|
import gameJSON from "../game.json";
|
||||||
import { mapState } from "vuex";
|
import { mapState } from "vuex";
|
||||||
import Countdown from "./Countdown";
|
import Countdown from "./Countdown.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: {
|
components: {
|
||||||
Countdown,
|
Countdown,
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
|
logoPath: () => `../assets/logos/${this?.edition?.id ?? "custom"}.png`,
|
||||||
teams: function () {
|
teams: function () {
|
||||||
const { players } = this.$store.state.players;
|
const { players } = this.$store.state.players;
|
||||||
const nonTravelers = this.$store.getters["players/nonTravelers"];
|
const nonTravelers = this.$store.getters["players/nonTravelers"];
|
||||||
|
|
|
@ -192,10 +192,10 @@
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapGetters, mapState } from "vuex";
|
import { mapGetters, mapState } from "vuex";
|
||||||
import Player from "./Player";
|
import Player from "./Player.vue";
|
||||||
import Token from "./Token";
|
import Token from "./Token.vue";
|
||||||
import ReminderModal from "./modals/ReminderModal";
|
import ReminderModal from "./modals/ReminderModal.vue";
|
||||||
import RoleModal from "./modals/RoleModal";
|
import RoleModal from "./modals/RoleModal.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: {
|
components: {
|
||||||
|
|
|
@ -169,7 +169,7 @@
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapGetters, mapState } from "vuex";
|
import { mapGetters, mapState } from "vuex";
|
||||||
import Countdown from "./Countdown";
|
import Countdown from "./Countdown.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: {
|
components: {
|
||||||
|
|
|
@ -42,9 +42,7 @@
|
||||||
class="edition"
|
class="edition"
|
||||||
:class="['edition-' + edition.id]"
|
:class="['edition-' + edition.id]"
|
||||||
:style="{
|
:style="{
|
||||||
backgroundImage: `url(${require(
|
backgroundImage: `url(${editionPath(edition)})`,
|
||||||
'../../assets/logos/' + edition.id + '.png',
|
|
||||||
)})`,
|
|
||||||
}"
|
}"
|
||||||
:key="edition.id"
|
:key="edition.id"
|
||||||
@click="runEdition(edition)"
|
@click="runEdition(edition)"
|
||||||
|
@ -158,9 +156,9 @@
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import { rolesJSON } from "../../store/modules/locale";
|
import { rolesJSON } from "../../store/modules/locale";
|
||||||
import Token from "../Token";
|
import Token from "../Token.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: {
|
components: {
|
||||||
|
@ -186,8 +184,7 @@ export default {
|
||||||
methods: {
|
methods: {
|
||||||
initPool() {
|
initPool() {
|
||||||
console.log("init pool");
|
console.log("init pool");
|
||||||
console.table(this.roles);
|
this.draftPool = rolesJSON.default;
|
||||||
this.draftPool = rolesJSON;
|
|
||||||
this.resetBuilt();
|
this.resetBuilt();
|
||||||
for (let [role] of this.roles) {
|
for (let [role] of this.roles) {
|
||||||
this.toggleRole(role);
|
this.toggleRole(role);
|
||||||
|
@ -327,6 +324,10 @@ export default {
|
||||||
// The editions contain no Fabled
|
// The editions contain no Fabled
|
||||||
this.$store.commit("players/setFabled", { fabled: [] });
|
this.$store.commit("players/setFabled", { fabled: [] });
|
||||||
},
|
},
|
||||||
|
editionPath(edition) {
|
||||||
|
return new URL(`../../assets/logos/${edition.id}.png`, import.meta.url)
|
||||||
|
.href;
|
||||||
|
},
|
||||||
...mapMutations(["toggleModal", "setEdition"]),
|
...mapMutations(["toggleModal", "setEdition"]),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -11,8 +11,8 @@
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import Token from "../Token";
|
import Token from "../Token.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: { Token, Modal },
|
components: { Token, Modal },
|
||||||
|
|
|
@ -23,7 +23,7 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
|
|
@ -56,11 +56,7 @@
|
||||||
backgroundImage: `url(${
|
backgroundImage: `url(${
|
||||||
role.image && grimoire.isImageOptIn
|
role.image && grimoire.isImageOptIn
|
||||||
? role.image
|
? role.image
|
||||||
: require(
|
: rolePath(role)
|
||||||
'../../assets/icons/' +
|
|
||||||
(role.imageAlt || role.id) +
|
|
||||||
'.png',
|
|
||||||
)
|
|
||||||
})`,
|
})`,
|
||||||
}"
|
}"
|
||||||
></span>
|
></span>
|
||||||
|
@ -83,11 +79,7 @@
|
||||||
backgroundImage: `url(${
|
backgroundImage: `url(${
|
||||||
role.image && grimoire.isImageOptIn
|
role.image && grimoire.isImageOptIn
|
||||||
? role.image
|
? role.image
|
||||||
: require(
|
: rolePath(role)
|
||||||
'../../assets/icons/' +
|
|
||||||
(role.imageAlt || role.id) +
|
|
||||||
'.png',
|
|
||||||
)
|
|
||||||
})`,
|
})`,
|
||||||
}"
|
}"
|
||||||
></span>
|
></span>
|
||||||
|
@ -127,7 +119,7 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
@ -274,6 +266,12 @@ export default {
|
||||||
...mapState("players", ["players", "fabled"]),
|
...mapState("players", ["players", "fabled"]),
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
rolePath(role) {
|
||||||
|
return new URL(
|
||||||
|
`../../assets/icons/${role.imageAlt || role.id}.png`,
|
||||||
|
import.meta.url,
|
||||||
|
).href;
|
||||||
|
},
|
||||||
...mapMutations(["toggleModal"]),
|
...mapMutations(["toggleModal"]),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -32,11 +32,7 @@
|
||||||
backgroundImage: `url(${
|
backgroundImage: `url(${
|
||||||
role.image && grimoire.isImageOptIn
|
role.image && grimoire.isImageOptIn
|
||||||
? role.image
|
? role.image
|
||||||
: require(
|
: rolePath(role)
|
||||||
'../../assets/icons/' +
|
|
||||||
(role.imageAlt || role.id) +
|
|
||||||
'.png',
|
|
||||||
)
|
|
||||||
})`,
|
})`,
|
||||||
}"
|
}"
|
||||||
></span>
|
></span>
|
||||||
|
@ -62,17 +58,13 @@
|
||||||
<span
|
<span
|
||||||
class="icon"
|
class="icon"
|
||||||
:style="{
|
:style="{
|
||||||
backgroundImage: `url(${require(
|
backgroundImage: rolePath(jinx.first),
|
||||||
'../../assets/icons/' + jinx.first.id + '.png',
|
|
||||||
)})`,
|
|
||||||
}"
|
}"
|
||||||
></span>
|
></span>
|
||||||
<span
|
<span
|
||||||
class="icon"
|
class="icon"
|
||||||
:style="{
|
:style="{
|
||||||
backgroundImage: `url(${require(
|
backgroundImage: rolePath(jinx.second),
|
||||||
'../../assets/icons/' + jinx.second.id + '.png',
|
|
||||||
)})`,
|
|
||||||
}"
|
}"
|
||||||
></span>
|
></span>
|
||||||
<div class="role">
|
<div class="role">
|
||||||
|
@ -91,7 +83,7 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
@ -147,6 +139,12 @@ export default {
|
||||||
...mapState("players", ["players"]),
|
...mapState("players", ["players"]),
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
rolePath(role) {
|
||||||
|
return new URL(
|
||||||
|
`../../assets/icons/${role.imageAlt || role.id}.png`,
|
||||||
|
import.meta.url,
|
||||||
|
).href;
|
||||||
|
},
|
||||||
...mapMutations(["toggleModal"]),
|
...mapMutations(["toggleModal"]),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -18,11 +18,7 @@
|
||||||
backgroundImage: `url(${
|
backgroundImage: `url(${
|
||||||
reminder.image && grimoire.isImageOptIn
|
reminder.image && grimoire.isImageOptIn
|
||||||
? reminder.image
|
? reminder.image
|
||||||
: require(
|
: rolePath(reminder.role)
|
||||||
'../../assets/icons/' +
|
|
||||||
(reminder.imageAlt || reminder.role) +
|
|
||||||
'.png',
|
|
||||||
)
|
|
||||||
})`,
|
})`,
|
||||||
}"
|
}"
|
||||||
></span>
|
></span>
|
||||||
|
@ -33,7 +29,7 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -137,6 +133,12 @@ export default {
|
||||||
});
|
});
|
||||||
this.$store.commit("toggleModal", "reminder");
|
this.$store.commit("toggleModal", "reminder");
|
||||||
},
|
},
|
||||||
|
rolePath(role) {
|
||||||
|
return new URL(
|
||||||
|
`../assets/icons/${role.imageAlt || role.id}.png`,
|
||||||
|
import.meta.url,
|
||||||
|
).href;
|
||||||
|
},
|
||||||
...mapMutations(["toggleModal"]),
|
...mapMutations(["toggleModal"]),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -50,8 +50,8 @@
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import Token from "../Token";
|
import Token from "../Token.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: { Token, Modal },
|
components: { Token, Modal },
|
||||||
|
|
|
@ -77,9 +77,9 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import gameJSON from "./../../game";
|
import gameJSON from "../../game.json";
|
||||||
import Token from "./../Token";
|
import Token from "../Token.vue";
|
||||||
import { mapGetters, mapMutations, mapState } from "vuex";
|
import { mapGetters, mapMutations, mapState } from "vuex";
|
||||||
|
|
||||||
const randomElement = (arr) => arr[Math.floor(Math.random() * arr.length)];
|
const randomElement = (arr) => arr[Math.floor(Math.random() * arr.length)];
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: {
|
components: {
|
||||||
|
|
|
@ -88,7 +88,7 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal.vue";
|
||||||
import { mapMutations, mapState } from "vuex";
|
import { mapMutations, mapState } from "vuex";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import Vue from "vue";
|
import Vue from "vue";
|
||||||
import App from "./App";
|
import App from "./App.vue";
|
||||||
import store from "./store";
|
import store from "./store";
|
||||||
import { library } from "@fortawesome/fontawesome-svg-core";
|
import { library } from "@fortawesome/fontawesome-svg-core";
|
||||||
import { fas } from "@fortawesome/free-solid-svg-icons";
|
import { fas } from "@fortawesome/free-solid-svg-icons";
|
||||||
|
|
|
@ -1,290 +1,272 @@
|
||||||
import Vue from "vue";
|
import Vue from "vue";
|
||||||
import Vuex from "vuex";
|
import Vuex from "vuex";
|
||||||
import persistence from "./persistence";
|
import persistence from "./persistence.js";
|
||||||
import socket from "./socket";
|
import socket from "./socket.js";
|
||||||
import players from "./modules/players";
|
import players from "./modules/players.js";
|
||||||
import session from "./modules/session";
|
import session from "./modules/session.js";
|
||||||
import editionJSON from "../editions.json";
|
import editionJSON from "../editions.json";
|
||||||
|
|
||||||
import { locale, rolesJSON, jinxesJSON, fabledJSON } from "./modules/locale";
|
|
||||||
|
|
||||||
Vue.use(Vuex);
|
Vue.use(Vuex);
|
||||||
|
|
||||||
// helper functions
|
const set = (key) => (state, value) => {
|
||||||
const getRolesByEdition = (edition = editionJSON.official[0]) => {
|
state.grimoire[key] = value;
|
||||||
return new Map(
|
|
||||||
rolesJSON
|
|
||||||
.filter((r) => r.edition === edition.id || edition.roles.includes(r.id))
|
|
||||||
.sort((a, b) => b.team.localeCompare(a.team))
|
|
||||||
.map((role) => [role.id, role]),
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const getTravelersNotInEdition = (edition = editionJSON.official[0]) => {
|
const toggle = (key) => (state) => {
|
||||||
return new Map(
|
state.grimoire[key] = !state.grimoire[key];
|
||||||
rolesJSON
|
|
||||||
.filter(
|
|
||||||
(r) =>
|
|
||||||
r.team === "traveler" &&
|
|
||||||
r.edition !== edition.id &&
|
|
||||||
!edition.roles.includes(r.id),
|
|
||||||
)
|
|
||||||
.map((role) => [role.id, role]),
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const set =
|
const loadLocale = async () => {
|
||||||
(key) =>
|
const { locale, rolesJSON, jinxesJSON, fabledJSON } = await import(
|
||||||
({ grimoire }, val) => {
|
"./modules/locale"
|
||||||
grimoire[key] = val;
|
);
|
||||||
|
return { locale, rolesJSON, jinxesJSON, fabledJSON };
|
||||||
|
};
|
||||||
|
|
||||||
|
const initializeStore = async () => {
|
||||||
|
const { locale, rolesJSON, jinxesJSON, fabledJSON } = await loadLocale();
|
||||||
|
|
||||||
|
const getRolesByEdition = (edition = editionJSON.official[0]) => {
|
||||||
|
return new Map(
|
||||||
|
rolesJSON.default
|
||||||
|
.filter((r) => r.edition === edition.id || edition.roles.includes(r.id))
|
||||||
|
.sort((a, b) => b.team.localeCompare(a.team))
|
||||||
|
.map((role) => [role.id, role]),
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const toggle =
|
const getTravelersNotInEdition = (edition = editionJSON.official[0]) => {
|
||||||
(key) =>
|
return new Map(
|
||||||
({ grimoire }, val) => {
|
rolesJSON.default
|
||||||
if (val === true || val === false) {
|
.filter(
|
||||||
grimoire[key] = val;
|
(r) =>
|
||||||
} else {
|
r.team === "traveler" &&
|
||||||
grimoire[key] = !grimoire[key];
|
r.edition !== edition.id &&
|
||||||
}
|
!edition.roles.includes(r.id),
|
||||||
|
)
|
||||||
|
.map((role) => [role.id, role]),
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const clean = (id) => id.toLocaleLowerCase().replace(/[^a-z0-9]/g, "");
|
const clean = (id) => id.toLocaleLowerCase().replace(/[^a-z0-9]/g, "");
|
||||||
|
|
||||||
// global data maps
|
const editionJSONbyId = new Map(
|
||||||
const editionJSONbyId = new Map(
|
editionJSON.official.map((edition) => [edition.id, edition]),
|
||||||
editionJSON.official.map((edition) => [edition.id, edition]),
|
|
||||||
);
|
|
||||||
const rolesJSONbyId = new Map(rolesJSON.map((role) => [role.id, role]));
|
|
||||||
const fabled = new Map(fabledJSON.map((role) => [role.id, role]));
|
|
||||||
|
|
||||||
// jinxes
|
|
||||||
let jinxes = {};
|
|
||||||
try {
|
|
||||||
// Note: can't fetch live list due to lack of CORS headers
|
|
||||||
// fetch("https://bloodontheclocktower.com/script/data/hatred.json")
|
|
||||||
// .then(res => res.json())
|
|
||||||
// .then(jinxesJSON => {
|
|
||||||
jinxes = new Map(
|
|
||||||
jinxesJSON.map(({ id, hatred }) => [
|
|
||||||
clean(id),
|
|
||||||
new Map(hatred.map(({ id, reason }) => [clean(id), reason])),
|
|
||||||
]),
|
|
||||||
);
|
);
|
||||||
// });
|
const rolesJSONbyId = new Map(
|
||||||
} catch (e) {
|
rolesJSON.default.map((role) => [role.id, role]),
|
||||||
console.error("couldn't load jinxes", e);
|
);
|
||||||
}
|
const fabled = new Map(fabledJSON.default.map((role) => [role.id, role]));
|
||||||
|
|
||||||
// base definition for custom roles
|
// jinxes
|
||||||
const customRole = {
|
let jinxes = {};
|
||||||
id: "",
|
try {
|
||||||
name: "",
|
jinxes = new Map(
|
||||||
image: "",
|
jinxesJSON.default.map(({ id, hatred }) => [
|
||||||
ability: "",
|
clean(id),
|
||||||
edition: "custom",
|
new Map(hatred.map(({ id, reason }) => [clean(id), reason])),
|
||||||
firstNight: 0,
|
]),
|
||||||
firstNightReminder: "",
|
);
|
||||||
otherNight: 0,
|
} catch (e) {
|
||||||
otherNightReminder: "",
|
console.error("couldn't load jinxes", e);
|
||||||
reminders: [],
|
}
|
||||||
remindersGlobal: [],
|
|
||||||
setup: false,
|
|
||||||
team: "townsfolk",
|
|
||||||
isCustom: true,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default new Vuex.Store({
|
// base definition for custom roles
|
||||||
modules: {
|
const customRole = {
|
||||||
players,
|
id: "",
|
||||||
session,
|
name: "",
|
||||||
},
|
image: "",
|
||||||
state: {
|
ability: "",
|
||||||
grimoire: {
|
edition: "custom",
|
||||||
isNight: false,
|
firstNight: 0,
|
||||||
isNightOrder: false,
|
firstNightReminder: "",
|
||||||
isRinging: false,
|
otherNight: 0,
|
||||||
isPublic: true,
|
otherNightReminder: "",
|
||||||
isMenuOpen: false,
|
reminders: [],
|
||||||
isStatic: false,
|
remindersGlobal: [],
|
||||||
isMuted: false,
|
setup: false,
|
||||||
isImageOptIn: false,
|
team: "townsfolk",
|
||||||
isStreamerMode: false,
|
isCustom: true,
|
||||||
isOrganVoteMode: false,
|
};
|
||||||
zoom: 0,
|
|
||||||
background: "",
|
return new Vuex.Store({
|
||||||
timer: {
|
modules: {
|
||||||
name: "",
|
players,
|
||||||
duration: 0,
|
session,
|
||||||
|
},
|
||||||
|
state: {
|
||||||
|
grimoire: {
|
||||||
|
isNight: false,
|
||||||
|
isNightOrder: false,
|
||||||
|
isRinging: false,
|
||||||
|
isPublic: true,
|
||||||
|
isMenuOpen: false,
|
||||||
|
isStatic: false,
|
||||||
|
isMuted: false,
|
||||||
|
isImageOptIn: false,
|
||||||
|
isStreamerMode: false,
|
||||||
|
isOrganVoteMode: false,
|
||||||
|
zoom: 0,
|
||||||
|
background: "",
|
||||||
|
timer: {
|
||||||
|
name: "",
|
||||||
|
duration: 0,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
modals: {
|
||||||
|
edition: false,
|
||||||
|
fabled: false,
|
||||||
|
gameState: false,
|
||||||
|
nightOrder: false,
|
||||||
|
reference: false,
|
||||||
|
reminder: false,
|
||||||
|
role: false,
|
||||||
|
roles: false,
|
||||||
|
voteHistory: false,
|
||||||
|
specialVote: false,
|
||||||
|
},
|
||||||
|
edition: editionJSONbyId.get("tb"),
|
||||||
|
editions: editionJSON,
|
||||||
|
roles: getRolesByEdition(),
|
||||||
|
otherTravelers: getTravelersNotInEdition(),
|
||||||
|
fabled,
|
||||||
|
jinxes,
|
||||||
|
locale,
|
||||||
},
|
},
|
||||||
modals: {
|
getters: {
|
||||||
edition: false,
|
customRolesStripped: ({ roles }) => {
|
||||||
fabled: false,
|
const customRoles = [];
|
||||||
gameState: false,
|
const customKeys = Object.keys(customRole);
|
||||||
nightOrder: false,
|
const strippedProps = [
|
||||||
reference: false,
|
"firstNightReminder",
|
||||||
reminder: false,
|
"otherNightReminder",
|
||||||
role: false,
|
"isCustom",
|
||||||
roles: false,
|
];
|
||||||
voteHistory: false,
|
roles.forEach((role) => {
|
||||||
specialVote: false,
|
if (!role.isCustom) {
|
||||||
},
|
customRoles.push({ id: role.id });
|
||||||
edition: editionJSONbyId.get("tb"),
|
} else {
|
||||||
editions: editionJSON,
|
const strippedRole = {};
|
||||||
roles: getRolesByEdition(),
|
|
||||||
otherTravelers: getTravelersNotInEdition(),
|
|
||||||
fabled,
|
|
||||||
jinxes,
|
|
||||||
locale,
|
|
||||||
},
|
|
||||||
getters: {
|
|
||||||
/**
|
|
||||||
* Return all custom roles, with default values and non-essential data stripped.
|
|
||||||
* Role object keys will be replaced with a numerical index to conserve bandwidth.
|
|
||||||
* @param roles
|
|
||||||
* @returns {[]}
|
|
||||||
*/
|
|
||||||
customRolesStripped: ({ roles }) => {
|
|
||||||
const customRoles = [];
|
|
||||||
const customKeys = Object.keys(customRole);
|
|
||||||
const strippedProps = [
|
|
||||||
"firstNightReminder",
|
|
||||||
"otherNightReminder",
|
|
||||||
"isCustom",
|
|
||||||
];
|
|
||||||
roles.forEach((role) => {
|
|
||||||
if (!role.isCustom) {
|
|
||||||
customRoles.push({ id: role.id });
|
|
||||||
} else {
|
|
||||||
const strippedRole = {};
|
|
||||||
for (let prop in role) {
|
|
||||||
if (strippedProps.includes(prop)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
const value = role[prop];
|
|
||||||
if (customKeys.includes(prop) && value !== customRole[prop]) {
|
|
||||||
strippedRole[customKeys.indexOf(prop)] = value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
customRoles.push(strippedRole);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return customRoles;
|
|
||||||
},
|
|
||||||
rolesJSONbyId: () => rolesJSONbyId,
|
|
||||||
},
|
|
||||||
mutations: {
|
|
||||||
setZoom: set("zoom"),
|
|
||||||
setBackground: set("background"),
|
|
||||||
toggleMuted: toggle("isMuted"),
|
|
||||||
toggleMenu: toggle("isMenuOpen"),
|
|
||||||
toggleNightOrder: toggle("isNightOrder"),
|
|
||||||
toggleStatic: toggle("isStatic"),
|
|
||||||
toggleNight: toggle("isNight"),
|
|
||||||
toggleRinging: toggle("isRinging"),
|
|
||||||
toggleGrimoire: toggle("isPublic"),
|
|
||||||
toggleImageOptIn: toggle("isImageOptIn"),
|
|
||||||
toggleStreamerMode: toggle("isStreamerMode"),
|
|
||||||
toggleOrganVoteMode: toggle("isOrganVoteMode"),
|
|
||||||
setTimer(state, timer) {
|
|
||||||
state.grimoire.timer = timer;
|
|
||||||
},
|
|
||||||
toggleModal({ modals }, name) {
|
|
||||||
if (name) {
|
|
||||||
modals[name] = !modals[name];
|
|
||||||
}
|
|
||||||
for (let modal in modals) {
|
|
||||||
if (modal === name) continue;
|
|
||||||
modals[modal] = false;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
/**
|
|
||||||
* Store custom roles
|
|
||||||
* @param state
|
|
||||||
* @param roles Array of role IDs or full role definitions
|
|
||||||
*/
|
|
||||||
setCustomRoles(state, roles) {
|
|
||||||
const processedRoles = roles
|
|
||||||
// replace numerical role object keys with matching key names
|
|
||||||
.map((role) => {
|
|
||||||
if (role[0]) {
|
|
||||||
const customKeys = Object.keys(customRole);
|
|
||||||
const mappedRole = {};
|
|
||||||
for (let prop in role) {
|
for (let prop in role) {
|
||||||
if (customKeys[prop]) {
|
if (strippedProps.includes(prop)) {
|
||||||
mappedRole[customKeys[prop]] = role[prop];
|
continue;
|
||||||
|
}
|
||||||
|
const value = role[prop];
|
||||||
|
if (customKeys.includes(prop) && value !== customRole[prop]) {
|
||||||
|
strippedRole[customKeys.indexOf(prop)] = value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return mappedRole;
|
customRoles.push(strippedRole);
|
||||||
} else {
|
|
||||||
return role;
|
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
// clean up role.id
|
return customRoles;
|
||||||
.map((role) => {
|
},
|
||||||
role.id = clean(role.id);
|
rolesJSONbyId: () => rolesJSONbyId,
|
||||||
return role;
|
},
|
||||||
})
|
mutations: {
|
||||||
// map existing roles to base definition or pre-populate custom roles to ensure all properties
|
setZoom: set("zoom"),
|
||||||
.map(
|
setBackground: set("background"),
|
||||||
(role) =>
|
toggleMuted: toggle("isMuted"),
|
||||||
rolesJSONbyId.get(role.id) ||
|
toggleMenu: toggle("isMenuOpen"),
|
||||||
state.roles.get(role.id) ||
|
toggleNightOrder: toggle("isNightOrder"),
|
||||||
Object.assign({}, customRole, role),
|
toggleStatic: toggle("isStatic"),
|
||||||
)
|
toggleNight: toggle("isNight"),
|
||||||
// default empty icons and placeholders, clean up firstNight / otherNight
|
toggleRinging: toggle("isRinging"),
|
||||||
.map((role) => {
|
toggleGrimoire: toggle("isPublic"),
|
||||||
if (rolesJSONbyId.get(role.id)) return role;
|
toggleImageOptIn: toggle("isImageOptIn"),
|
||||||
role.imageAlt = // map team to generic icon
|
toggleStreamerMode: toggle("isStreamerMode"),
|
||||||
{
|
toggleOrganVoteMode: toggle("isOrganVoteMode"),
|
||||||
townsfolk: "townsfolk",
|
setTimer(state, timer) {
|
||||||
outsider: "outsider",
|
state.grimoire.timer = timer;
|
||||||
minion: "minion",
|
},
|
||||||
demon: "demon",
|
toggleModal({ modals }, name) {
|
||||||
fabled: "fabled",
|
if (name) {
|
||||||
traveler: "traveler",
|
modals[name] = !modals[name];
|
||||||
}[role.team] || "custom";
|
}
|
||||||
role.firstNight = Math.abs(role.firstNight);
|
for (let modal in modals) {
|
||||||
role.otherNight = Math.abs(role.otherNight);
|
if (modal === name) continue;
|
||||||
return role;
|
modals[modal] = false;
|
||||||
})
|
}
|
||||||
// filter out roles that don't match an existing role and also don't have name/ability/team
|
},
|
||||||
.filter((role) => role.name && role.ability && role.team)
|
setCustomRoles(state, roles) {
|
||||||
// sort by team
|
const processedRoles = roles
|
||||||
.sort((a, b) => b.team.localeCompare(a.team));
|
.map((role) => {
|
||||||
// convert to Map without Fabled
|
if (role[0]) {
|
||||||
state.roles = new Map(
|
const customKeys = Object.keys(customRole);
|
||||||
processedRoles
|
const mappedRole = {};
|
||||||
.filter((role) => role.team !== "fabled")
|
for (let prop in role) {
|
||||||
.map((role) => [role.id, role]),
|
if (customKeys[prop]) {
|
||||||
);
|
mappedRole[customKeys[prop]] = role[prop];
|
||||||
// update Fabled to include custom Fabled from this script
|
}
|
||||||
state.fabled = new Map([
|
}
|
||||||
...processedRoles
|
return mappedRole;
|
||||||
.filter((r) => r.team === "fabled")
|
} else {
|
||||||
.map((r) => [r.id, r]),
|
return role;
|
||||||
...fabledJSON.map((role) => [role.id, role]),
|
}
|
||||||
]);
|
})
|
||||||
// update extraTravelers map to only show travelers not in this script
|
.map((role) => {
|
||||||
state.otherTravelers = new Map(
|
role.id = clean(role.id);
|
||||||
rolesJSON
|
return role;
|
||||||
.filter(
|
})
|
||||||
(r) => r.team === "traveler" && !roles.some((i) => i.id === r.id),
|
.map(
|
||||||
|
(role) =>
|
||||||
|
rolesJSONbyId.get(role.id) ||
|
||||||
|
state.roles.get(role.id) || { ...customRole, ...role },
|
||||||
)
|
)
|
||||||
.map((role) => [role.id, role]),
|
.map((role) => {
|
||||||
);
|
if (rolesJSONbyId.get(role.id)) return role;
|
||||||
|
role.imageAlt =
|
||||||
|
{
|
||||||
|
townsfolk: "townsfolk",
|
||||||
|
outsider: "outsider",
|
||||||
|
minion: "minion",
|
||||||
|
demon: "demon",
|
||||||
|
fabled: "fabled",
|
||||||
|
traveler: "traveler",
|
||||||
|
}[role.team] || "custom";
|
||||||
|
role.firstNight = Math.abs(role.firstNight);
|
||||||
|
role.otherNight = Math.abs(role.otherNight);
|
||||||
|
return role;
|
||||||
|
})
|
||||||
|
.filter((role) => role.name && role.ability && role.team)
|
||||||
|
.sort((a, b) => b.team.localeCompare(a.team));
|
||||||
|
|
||||||
|
state.roles = new Map(
|
||||||
|
processedRoles
|
||||||
|
.filter((role) => role.team !== "fabled")
|
||||||
|
.map((role) => [role.id, role]),
|
||||||
|
);
|
||||||
|
|
||||||
|
state.fabled = new Map([
|
||||||
|
...processedRoles
|
||||||
|
.filter((r) => r.team === "fabled")
|
||||||
|
.map((r) => [r.id, r]),
|
||||||
|
...fabledJSON.default.map((role) => [role.id, role]),
|
||||||
|
]);
|
||||||
|
|
||||||
|
state.otherTravelers = new Map(
|
||||||
|
rolesJSON.default
|
||||||
|
.filter(
|
||||||
|
(r) => r.team === "traveler" && !roles.some((i) => i.id === r.id),
|
||||||
|
)
|
||||||
|
.map((role) => [role.id, role]),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
setEdition(state, edition) {
|
||||||
|
if (editionJSONbyId.has(edition.id)) {
|
||||||
|
state.edition = editionJSONbyId.get(edition.id);
|
||||||
|
state.roles = getRolesByEdition(state.edition);
|
||||||
|
state.otherTravelers = getTravelersNotInEdition(state.edition);
|
||||||
|
} else {
|
||||||
|
state.edition = edition;
|
||||||
|
}
|
||||||
|
state.modals.edition = false;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
setEdition(state, edition) {
|
plugins: [persistence, socket],
|
||||||
if (editionJSONbyId.has(edition.id)) {
|
});
|
||||||
state.edition = editionJSONbyId.get(edition.id);
|
};
|
||||||
state.roles = getRolesByEdition(state.edition);
|
|
||||||
state.otherTravelers = getTravelersNotInEdition(state.edition);
|
// Create the store and export it
|
||||||
} else {
|
const store = await initializeStore();
|
||||||
state.edition = edition;
|
export default store;
|
||||||
}
|
|
||||||
state.modals.edition = false;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
plugins: [persistence, socket],
|
|
||||||
});
|
|
||||||
|
|
|
@ -23,7 +23,7 @@ if (!usedLanguage) {
|
||||||
usedLanguage = MASTER_LANGUAGE; // set to master language if no language is supported by both the user and the application
|
usedLanguage = MASTER_LANGUAGE; // set to master language if no language is supported by both the user and the application
|
||||||
}
|
}
|
||||||
|
|
||||||
export const locale = require(`../locale/${usedLanguage}/ui.json`);
|
export const locale = await import(`../locale/${usedLanguage}/ui.json`);
|
||||||
export const rolesJSON = require(`../locale/${usedLanguage}/roles.json`);
|
export const rolesJSON = await import(`../locale/${usedLanguage}/roles.json`);
|
||||||
export const jinxesJSON = require(`../locale/${usedLanguage}/hatred.json`);
|
export const jinxesJSON = await import(`../locale/${usedLanguage}/hatred.json`);
|
||||||
export const fabledJSON = require(`../locale/${usedLanguage}/fabled.json`);
|
export const fabledJSON = await import(`../locale/${usedLanguage}/fabled.json`);
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
module.exports = (store) => {
|
export default (store) => {
|
||||||
const updatePagetitle = (isPublic) =>
|
const updatePagetitle = (isPublic) =>
|
||||||
(document.title = `Blood on the Clocktower ${
|
(document.title = `Blood on the Clocktower ${
|
||||||
isPublic ? "Town Square" : "Grimoire"
|
isPublic ? "Town Square" : "Grimoire"
|
||||||
|
|
17
vite.config.js
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
import { defineConfig } from "vite";
|
||||||
|
import vue from "@vitejs/plugin-vue2";
|
||||||
|
|
||||||
|
export default defineConfig({
|
||||||
|
plugins: [vue()],
|
||||||
|
base: "/",
|
||||||
|
build: {
|
||||||
|
rollupOptions: {
|
||||||
|
output: {
|
||||||
|
entryFileNames: "[name].js",
|
||||||
|
chunkFileNames: "[name].[hash].js",
|
||||||
|
assetFileNames: "[name].[hash].[extname]",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
target: "es2022",
|
||||||
|
},
|
||||||
|
});
|
|
@ -1,12 +0,0 @@
|
||||||
const MiniCssExtractPlugin = require("mini-css-extract-plugin");
|
|
||||||
module.exports = {
|
|
||||||
configureWebpack: {
|
|
||||||
plugins: [
|
|
||||||
new MiniCssExtractPlugin({
|
|
||||||
filename: "[name].css",
|
|
||||||
chunkFilename: "[id].css",
|
|
||||||
}),
|
|
||||||
],
|
|
||||||
},
|
|
||||||
publicPath: "/",
|
|
||||||
};
|
|