This commit is contained in:
Pingumask 2023-11-10 23:28:33 +01:00
commit 080296e0aa
40 changed files with 699 additions and 620 deletions

View file

@ -1,16 +1,16 @@
module.exports = {
root: true,
env: {
node: true
node: true,
},
extends: ["plugin:vue/essential", "eslint:recommended", "@vue/prettier"],
parserOptions: {
ecmaVersion: 2020
ecmaVersion: 2020,
},
rules: {
"no-console": process.env.NODE_ENV === "production" ? "warn" : "off",
"no-debugger": process.env.NODE_ENV === "production" ? "warn" : "off",
"vue/multi-word-component-names": "off",
"vue/no-reserved-component-names": "off",
}
},
};

View file

@ -1,11 +1,17 @@
# Release Notes
---
## Upcomming Version
### Version 3.17.0
- Updating english jinxes
- Adding an asterisk in the roles reference
- Correcting the message when someone wants to exile a Traveller
- Correcting Leviathan's english description
- Correcting "Late Night Drive By"'s name
- Adding a token "Used" to the Doomsayer
- Updated packages & Dockerfile for node >=18
### Version 3.16.0 (merged upstream 2.16.2)
- fixed custom script format to support new script tool JSON
- updated packages to be compatible with Node >= 18 again

View file

@ -1,10 +1,11 @@
FROM node:16
FROM node:18
RUN apt update && apt install -y\
git
WORKDIR /app
git\
&& apt clean
WORKDIR /app/townsquare
COPY package*.json .
RUN npm install
# npm rebuild avoids having misconfigurations if npm install has been run in the folder from windows before building the docker image
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 . .
CMD ["npm","run","serve"]

View file

@ -6,12 +6,15 @@ services:
restart: unless-stopped
volumes:
- "${SSH_KEYS}:/root/.ssh"
- botc_front:/app/townsquare
environment:
NODE_ENV: development
ports:
- "${NODE_PORT}:8080"
working_dir: /app
working_dir: /app/townsquare
logging:
driver: "json-file"
options:
max-size: "200k"
volumes:
botc_front:

194
package-lock.json generated
View file

@ -1,12 +1,12 @@
{
"name": "townsquare",
"version": "3.16.0",
"version": "3.17.0",
"lockfileVersion": 3,
"requires": true,
"packages": {
"": {
"name": "townsquare",
"version": "3.15.0",
"version": "3.17.0",
"license": "GPL-3.0",
"dependencies": {
"@fortawesome/fontawesome-svg-core": "^1.2.32",
@ -29,6 +29,9 @@
"eslint-plugin-prettier": "^5.0.1",
"eslint-plugin-vue": "^9.18.1",
"prettier": "^3.0.3"
},
"engines": {
"node": "^16"
}
},
"node_modules/@aashutoshrathi/word-wrap": {
@ -130,9 +133,9 @@
}
},
"node_modules/@babel/compat-data": {
"version": "7.23.2",
"resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.23.2.tgz",
"integrity": "sha512-0S9TQMmDHlqAZ2ITT95irXKfxN9bncq8ZCoJhun3nHL/lLUxd2NKBJYoNGWH7S0hz6fRQwWlAWn/ILM0C70KZQ==",
"version": "7.23.3",
"resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.23.3.tgz",
"integrity": "sha512-BmR4bWbDIoFJmJ9z2cZ8Gmm2MXgEDgjdWgpKmKWUt54UGFJdlj31ECtbaDvCG/qVdG3AQ1SfpZEs01lUFbzLOQ==",
"engines": {
"node": ">=6.9.0"
}
@ -246,9 +249,9 @@
}
},
"node_modules/@babel/parser": {
"version": "7.23.0",
"resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.23.0.tgz",
"integrity": "sha512-vvPKKdMemU85V9WE/l5wZEmImpCtLqbnTvqDS2U1fJ96KrxoW7KrXhNsNCblQlg8Ck4b85yxdTyelsMUgFUXiw==",
"version": "7.23.3",
"resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.23.3.tgz",
"integrity": "sha512-uVsWNvlVsIninV2prNz/3lHCb+5CJ+e+IUBfbjToAHODtfGYLfCFuY4AU7TskI+dAKk+njsPiBjq1gKTvZOBaw==",
"bin": {
"parser": "bin/babel-parser.js"
},
@ -624,66 +627,66 @@
}
},
"node_modules/@types/body-parser": {
"version": "1.19.4",
"resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.4.tgz",
"integrity": "sha512-N7UDG0/xiPQa2D/XrVJXjkWbpqHCd2sBaB32ggRF2l83RhPfamgKGF8gwwqyksS95qUS5ZYF9aF+lLPRlwI2UA==",
"version": "1.19.5",
"resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.5.tgz",
"integrity": "sha512-fB3Zu92ucau0iQ0JMCFQE7b/dv8Ot07NI3KaZIkIUNXq82k4eBAqUaneXfleGY9JWskeS9y+u0nXMyspcuQrCg==",
"dependencies": {
"@types/connect": "*",
"@types/node": "*"
}
},
"node_modules/@types/bonjour": {
"version": "3.5.12",
"resolved": "https://registry.npmjs.org/@types/bonjour/-/bonjour-3.5.12.tgz",
"integrity": "sha512-ky0kWSqXVxSqgqJvPIkgFkcn4C8MnRog308Ou8xBBIVo39OmUFy+jqNe0nPwLCDFxUpmT9EvT91YzOJgkDRcFg==",
"version": "3.5.13",
"resolved": "https://registry.npmjs.org/@types/bonjour/-/bonjour-3.5.13.tgz",
"integrity": "sha512-z9fJ5Im06zvUL548KvYNecEVlA7cVDkGUi6kZusb04mpyEFKCIZJvloCcmpmLaIahDpOQGHaHmG6imtPMmPXGQ==",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/connect": {
"version": "3.4.37",
"resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.37.tgz",
"integrity": "sha512-zBUSRqkfZ59OcwXon4HVxhx5oWCJmc0OtBTK05M+p0dYjgN6iTwIL2T/WbsQZrEsdnwaF9cWQ+azOnpPvIqY3Q==",
"version": "3.4.38",
"resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz",
"integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/connect-history-api-fallback": {
"version": "1.5.2",
"resolved": "https://registry.npmjs.org/@types/connect-history-api-fallback/-/connect-history-api-fallback-1.5.2.tgz",
"integrity": "sha512-gX2j9x+NzSh4zOhnRPSdPPmTepS4DfxES0AvIFv3jGv5QyeAJf6u6dY5/BAoAJU9Qq1uTvwOku8SSC2GnCRl6Q==",
"version": "1.5.3",
"resolved": "https://registry.npmjs.org/@types/connect-history-api-fallback/-/connect-history-api-fallback-1.5.3.tgz",
"integrity": "sha512-6mfQ6iNvhSKCZJoY6sIG3m0pKkdUcweVNOLuBBKvoWGzl2yRxOJcYOTRyLKt3nxXvBLJWa6QkW//tgbIwJehmA==",
"dependencies": {
"@types/express-serve-static-core": "*",
"@types/node": "*"
}
},
"node_modules/@types/eslint": {
"version": "8.44.6",
"resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-8.44.6.tgz",
"integrity": "sha512-P6bY56TVmX8y9J87jHNgQh43h6VVU+6H7oN7hgvivV81K2XY8qJZ5vqPy/HdUoVIelii2kChYVzQanlswPWVFw==",
"version": "8.44.7",
"resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-8.44.7.tgz",
"integrity": "sha512-f5ORu2hcBbKei97U73mf+l9t4zTGl74IqZ0GQk4oVea/VS8tQZYkUveSYojk+frraAVYId0V2WC9O4PTNru2FQ==",
"dependencies": {
"@types/estree": "*",
"@types/json-schema": "*"
}
},
"node_modules/@types/eslint-scope": {
"version": "3.7.6",
"resolved": "https://registry.npmjs.org/@types/eslint-scope/-/eslint-scope-3.7.6.tgz",
"integrity": "sha512-zfM4ipmxVKWdxtDaJ3MP3pBurDXOCoyjvlpE3u6Qzrmw4BPbfm4/ambIeTk/r/J0iq/+2/xp0Fmt+gFvXJY2PQ==",
"version": "3.7.7",
"resolved": "https://registry.npmjs.org/@types/eslint-scope/-/eslint-scope-3.7.7.tgz",
"integrity": "sha512-MzMFlSLBqNF2gcHWO0G1vP/YQyfvrxZ0bF+u7mzUdZ1/xK4A4sru+nraZz5i3iEIk1l1uyicaDVTB4QbbEkAYg==",
"dependencies": {
"@types/eslint": "*",
"@types/estree": "*"
}
},
"node_modules/@types/estree": {
"version": "1.0.4",
"resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.4.tgz",
"integrity": "sha512-2JwWnHK9H+wUZNorf2Zr6ves96WHoWDJIftkcxPKsS7Djta6Zu519LarhRNljPXkpsZR2ZMwNCPeW7omW07BJw=="
"version": "1.0.5",
"resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz",
"integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw=="
},
"node_modules/@types/express": {
"version": "4.17.20",
"resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.20.tgz",
"integrity": "sha512-rOaqlkgEvOW495xErXMsmyX3WKBInbhG5eqojXYi3cGUaLoRDlXa5d52fkfWZT963AZ3v2eZ4MbKE6WpDAGVsw==",
"version": "4.17.21",
"resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.21.tgz",
"integrity": "sha512-ejlPM315qwLpaQlQDTjPdsUFSc6ZsP4AN6AlWnogPjQ7CVi7PYF3YVz+CY3jE2pwYf7E/7HlDAN0rV2GxTG0HQ==",
"dependencies": {
"@types/body-parser": "*",
"@types/express-serve-static-core": "^4.17.33",
@ -692,9 +695,9 @@
}
},
"node_modules/@types/express-serve-static-core": {
"version": "4.17.39",
"resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.39.tgz",
"integrity": "sha512-BiEUfAiGCOllomsRAZOiMFP7LAnrifHpt56pc4Z7l9K6ACyN06Ns1JLMBxwkfLOjJRlSf06NwWsT7yzfpaVpyQ==",
"version": "4.17.41",
"resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.41.tgz",
"integrity": "sha512-OaJ7XLaelTgrvlZD8/aa0vvvxZdUmlCn6MtWeB7TkiKW70BQLc9XEPpDLPdbo52ZhXUCrznlWdCHWxJWtdyajA==",
"dependencies": {
"@types/node": "*",
"@types/qs": "*",
@ -708,68 +711,68 @@
"integrity": "sha512-oh/6byDPnL1zeNXFrDXFLyZjkr1MsBG667IM792caf1L2UPOOMf65NFzjUH/ltyfwjAGfs1rsX1eftK0jC/KIg=="
},
"node_modules/@types/http-errors": {
"version": "2.0.3",
"resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.3.tgz",
"integrity": "sha512-pP0P/9BnCj1OVvQR2lF41EkDG/lWWnDyA203b/4Fmi2eTyORnBtcDoKDwjWQthELrBvWkMOrvSOnZ8OVlW6tXA=="
"version": "2.0.4",
"resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.4.tgz",
"integrity": "sha512-D0CFMMtydbJAegzOyHjtiKPLlvnm3iTZyZRSZoLq2mRhDdmLfIWOCYPfQJ4cu2erKghU++QvjcUjp/5h7hESpA=="
},
"node_modules/@types/http-proxy": {
"version": "1.17.13",
"resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.13.tgz",
"integrity": "sha512-GkhdWcMNiR5QSQRYnJ+/oXzu0+7JJEPC8vkWXK351BkhjraZF+1W13CUYARUvX9+NqIU2n6YHA4iwywsc/M6Sw==",
"version": "1.17.14",
"resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.14.tgz",
"integrity": "sha512-SSrD0c1OQzlFX7pGu1eXxSEjemej64aaNPRhhVYUGqXh0BtldAAx37MG8btcumvpgKyZp1F5Gn3JkktdxiFv6w==",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/json-schema": {
"version": "7.0.14",
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.14.tgz",
"integrity": "sha512-U3PUjAudAdJBeC2pgN8uTIKgxrb4nlDF3SF0++EldXQvQBGkpFZMSnwQiIoDU77tv45VgNkl/L4ouD+rEomujw=="
"version": "7.0.15",
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz",
"integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA=="
},
"node_modules/@types/mime": {
"version": "1.3.4",
"resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.4.tgz",
"integrity": "sha512-1Gjee59G25MrQGk8bsNvC6fxNiRgUlGn2wlhGf95a59DrprnnHk80FIMMFG9XHMdrfsuA119ht06QPDXA1Z7tw=="
"version": "1.3.5",
"resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.5.tgz",
"integrity": "sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w=="
},
"node_modules/@types/minimist": {
"version": "1.2.4",
"resolved": "https://registry.npmjs.org/@types/minimist/-/minimist-1.2.4.tgz",
"integrity": "sha512-Kfe/D3hxHTusnPNRbycJE1N77WHDsdS4AjUYIzlDzhDrS47NrwuL3YW4VITxwR7KCVpzwgy4Rbj829KSSQmwXQ=="
"version": "1.2.5",
"resolved": "https://registry.npmjs.org/@types/minimist/-/minimist-1.2.5.tgz",
"integrity": "sha512-hov8bUuiLiyFPGyFPE1lwWhmzYbirOXQNNo40+y3zow8aFVTeyn3VWL0VFFfdNddA8S4Vf0Tc062rzyNr7Paag=="
},
"node_modules/@types/node": {
"version": "20.8.10",
"resolved": "https://registry.npmjs.org/@types/node/-/node-20.8.10.tgz",
"integrity": "sha512-TlgT8JntpcbmKUFzjhsyhGfP2fsiz1Mv56im6enJ905xG1DAYesxJaeSbGqQmAw8OWPdhyJGhGSQGKRNJ45u9w==",
"version": "20.9.0",
"resolved": "https://registry.npmjs.org/@types/node/-/node-20.9.0.tgz",
"integrity": "sha512-nekiGu2NDb1BcVofVcEKMIwzlx4NjHlcjhoxxKBNLtz15Y1z7MYf549DFvkHSId02Ax6kGwWntIBPC3l/JZcmw==",
"dependencies": {
"undici-types": "~5.26.4"
}
},
"node_modules/@types/node-forge": {
"version": "1.3.8",
"resolved": "https://registry.npmjs.org/@types/node-forge/-/node-forge-1.3.8.tgz",
"integrity": "sha512-vGXshY9vim9CJjrpcS5raqSjEfKlJcWy2HNdgUasR66fAnVEYarrf1ULV4nfvpC1nZq/moA9qyqBcu83x+Jlrg==",
"version": "1.3.9",
"resolved": "https://registry.npmjs.org/@types/node-forge/-/node-forge-1.3.9.tgz",
"integrity": "sha512-meK88cx/sTalPSLSoCzkiUB4VPIFHmxtXm5FaaqRDqBX2i/Sy8bJ4odsan0b20RBjPh06dAQ+OTTdnyQyhJZyQ==",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/normalize-package-data": {
"version": "2.4.3",
"resolved": "https://registry.npmjs.org/@types/normalize-package-data/-/normalize-package-data-2.4.3.tgz",
"integrity": "sha512-ehPtgRgaULsFG8x0NeYJvmyH1hmlfsNLujHe9dQEia/7MAJYdzMSi19JtchUHjmBA6XC/75dK55mzZH+RyieSg=="
"version": "2.4.4",
"resolved": "https://registry.npmjs.org/@types/normalize-package-data/-/normalize-package-data-2.4.4.tgz",
"integrity": "sha512-37i+OaWTh9qeK4LSHPsyRC7NahnGotNuZvjLSgcPzblpHB3rrCJxAOgI5gCdKm7coonsaX1Of0ILiTcnZjbfxA=="
},
"node_modules/@types/parse-json": {
"version": "4.0.1",
"resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.1.tgz",
"integrity": "sha512-3YmXzzPAdOTVljVMkTMBdBEvlOLg2cDQaDhnnhT3nT9uDbnJzjWhKlzb+desT12Y7tGqaN6d+AbozcKzyL36Ng=="
"version": "4.0.2",
"resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.2.tgz",
"integrity": "sha512-dISoDXWWQwUquiKsyZ4Ng+HX2KsPL7LyHKHQwgGFEA3IaKac4Obd+h2a/a6waisAoepJlBcx9paWqjA8/HVjCw=="
},
"node_modules/@types/qs": {
"version": "6.9.9",
"resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.9.tgz",
"integrity": "sha512-wYLxw35euwqGvTDx6zfY1vokBFnsK0HNrzc6xNHchxfO2hpuRg74GbkEW7e3sSmPvj0TjCDT1VCa6OtHXnubsg=="
"version": "6.9.10",
"resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.10.tgz",
"integrity": "sha512-3Gnx08Ns1sEoCrWssEgTSJs/rsT2vhGP+Ja9cnnk9k4ALxinORlQneLXFeFKOTJMOeZUFD1s7w+w2AphTpvzZw=="
},
"node_modules/@types/range-parser": {
"version": "1.2.6",
"resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.6.tgz",
"integrity": "sha512-+0autS93xyXizIYiyL02FCY8N+KkKPhILhcUSA276HxzreZ16kl+cmwvV2qAM/PuCCwPXzOXOWhiPcw20uSFcA=="
"version": "1.2.7",
"resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.7.tgz",
"integrity": "sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ=="
},
"node_modules/@types/retry": {
"version": "0.12.0",
@ -777,26 +780,26 @@
"integrity": "sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA=="
},
"node_modules/@types/send": {
"version": "0.17.3",
"resolved": "https://registry.npmjs.org/@types/send/-/send-0.17.3.tgz",
"integrity": "sha512-/7fKxvKUoETxjFUsuFlPB9YndePpxxRAOfGC/yJdc9kTjTeP5kRCTzfnE8kPUKCeyiyIZu0YQ76s50hCedI1ug==",
"version": "0.17.4",
"resolved": "https://registry.npmjs.org/@types/send/-/send-0.17.4.tgz",
"integrity": "sha512-x2EM6TJOybec7c52BX0ZspPodMsQUd5L6PRwOunVyVUhXiBSKf3AezDL8Dgvgt5o0UfKNfuA0eMLr2wLT4AiBA==",
"dependencies": {
"@types/mime": "^1",
"@types/node": "*"
}
},
"node_modules/@types/serve-index": {
"version": "1.9.3",
"resolved": "https://registry.npmjs.org/@types/serve-index/-/serve-index-1.9.3.tgz",
"integrity": "sha512-4KG+yMEuvDPRrYq5fyVm/I2uqAJSAwZK9VSa+Zf+zUq9/oxSSvy3kkIqyL+jjStv6UCVi8/Aho0NHtB1Fwosrg==",
"version": "1.9.4",
"resolved": "https://registry.npmjs.org/@types/serve-index/-/serve-index-1.9.4.tgz",
"integrity": "sha512-qLpGZ/c2fhSs5gnYsQxtDEq3Oy8SXPClIXkW5ghvAvsNuVSA8k+gCONcUCS/UjLEYvYps+e8uBtfgXgvhwfNug==",
"dependencies": {
"@types/express": "*"
}
},
"node_modules/@types/serve-static": {
"version": "1.15.4",
"resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.4.tgz",
"integrity": "sha512-aqqNfs1XTF0HDrFdlY//+SGUxmdSUbjeRXb5iaZc3x0/vMbYmdw9qvOgHWOyyLFxSSRnUuP5+724zBgfw8/WAw==",
"version": "1.15.5",
"resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.5.tgz",
"integrity": "sha512-PDRk21MnK70hja/YF8AHfC7yIsiQHn1rcXx7ijCFBX/k+XQJhQT/gw3xekXKJvx+5SXaMMS8oqQy09Mzvz2TuQ==",
"dependencies": {
"@types/http-errors": "*",
"@types/mime": "*",
@ -804,17 +807,17 @@
}
},
"node_modules/@types/sockjs": {
"version": "0.3.35",
"resolved": "https://registry.npmjs.org/@types/sockjs/-/sockjs-0.3.35.tgz",
"integrity": "sha512-tIF57KB+ZvOBpAQwSaACfEu7htponHXaFzP7RfKYgsOS0NoYnn+9+jzp7bbq4fWerizI3dTB4NfAZoyeQKWJLw==",
"version": "0.3.36",
"resolved": "https://registry.npmjs.org/@types/sockjs/-/sockjs-0.3.36.tgz",
"integrity": "sha512-MK9V6NzAS1+Ud7JV9lJLFqW85VbC9dq3LmwZCuBe4wBDgKC0Kj/jd8Xl+nSviU+Qc3+m7umHHyHg//2KSa0a0Q==",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/ws": {
"version": "8.5.8",
"resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.8.tgz",
"integrity": "sha512-flUksGIQCnJd6sZ1l5dqCEG/ksaoAg/eUwiLAGTJQcfgvZJKF++Ta4bJA6A5aPSJmsr+xlseHn4KLgVlNnvPTg==",
"version": "8.5.9",
"resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.9.tgz",
"integrity": "sha512-jbdrY0a8lxfdTp/+r7Z4CkycbOFN8WX+IOchLJr3juT/xzbJ8URyTVSJ/hvNdadTgM1mnedb47n+Y31GsFnQlg==",
"dependencies": {
"@types/node": "*"
}
@ -2986,9 +2989,9 @@
"integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="
},
"node_modules/electron-to-chromium": {
"version": "1.4.576",
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.576.tgz",
"integrity": "sha512-yXsZyXJfAqzWk1WKryr0Wl0MN2D47xodPvEEwlVePBnhU5E7raevLQR+E6b9JAD3GfL/7MbAL9ZtWQQPcLx7wA=="
"version": "1.4.581",
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.581.tgz",
"integrity": "sha512-6uhqWBIapTJUxgPTCHH9sqdbxIMPt7oXl0VcAL1kOtlU6aECdcMncCrX5Z7sHQ/invtrC9jUQUef7+HhO8vVFw=="
},
"node_modules/emoji-regex": {
"version": "8.0.0",
@ -3056,9 +3059,9 @@
}
},
"node_modules/es-module-lexer": {
"version": "1.3.1",
"resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.3.1.tgz",
"integrity": "sha512-JUFAyicQV9mXc3YRxPnDlrfBKpqt6hUYzz9/boprUJHs4e4KVr3XwOF70doO6gwXUor6EWZJAyWAfKki84t20Q=="
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.4.0.tgz",
"integrity": "sha512-lcCr3v3OLezdfFyx9r5NRYHOUTQNnFEQ9E87Mx8Kc+iqyJNkO7MJoB4GQRTlIMw9kLLTwGw0OAkm4BQQud/d9g=="
},
"node_modules/escalade": {
"version": "3.1.1",
@ -3619,9 +3622,9 @@
"dev": true
},
"node_modules/fast-glob": {
"version": "3.3.1",
"resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.1.tgz",
"integrity": "sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg==",
"version": "3.3.2",
"resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz",
"integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==",
"dependencies": {
"@nodelib/fs.stat": "^2.0.2",
"@nodelib/fs.walk": "^1.2.3",
@ -8027,13 +8030,6 @@
"dependencies": {
"hash-sum": "^1.0.2",
"loader-utils": "^1.0.2"
},
"dependencies": {
"hash-sum": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/hash-sum/-/hash-sum-1.0.2.tgz",
"integrity": "sha512-fUs4B4L+mlt8/XAtSOGMUO1TXmAelItBPtJG7CyHJfYTdDjwisntGO2JQz7oUsatOY9o68+57eziUVNw/mRHmA=="
}
}
},
"node_modules/vue-style-loader/node_modules/hash-sum": {

View file

@ -1,6 +1,6 @@
{
"name": "townsquare",
"version": "3.16.0",
"version": "3.17.0",
"description": "Blood on the Clocktower Town Square",
"author": "Pingumaskt",
"scripts": {

View file

@ -5,11 +5,11 @@
tabindex="-1"
:class="{
night: grimoire.isNight,
static: grimoire.isStatic
static: grimoire.isStatic,
}"
:style="{
backgroundImage: `url('${background}')`,
backgroundColor: `${backgroundColor}`
backgroundColor: `${backgroundColor}`,
}"
>
<video
@ -70,24 +70,24 @@ export default {
Menu,
EditionModal,
RolesModal,
Gradients
Gradients,
},
computed: {
...mapState(["grimoire", "session", "edition"]),
...mapState("players", ["players"]),
background: function() {
background: function () {
if (this.grimoire.isStreamerMode) {
return "none";
}
return this.grimoire.background || this.edition.background || "none";
},
backgroundColor: function() {
backgroundColor: function () {
return this.grimoire.isStreamerMode ? "#00FF00" : "transparent";
}
},
},
data() {
return {
version
version,
};
},
methods: {
@ -136,8 +136,8 @@ export default {
case "escape":
this.$store.commit("toggleModal");
}
}
}
},
},
};
</script>
@ -147,7 +147,8 @@ export default {
@font-face {
font-family: "Papyrus";
src: url("assets/fonts/papyrus.eot"); /* IE9*/
src: url("assets/fonts/papyrus.eot?#iefix") format("embedded-opentype"),
src:
url("assets/fonts/papyrus.eot?#iefix") format("embedded-opentype"),
/* IE6-IE8 */ url("assets/fonts/papyrus.woff2") format("woff2"),
/* chrome firefox */ url("assets/fonts/papyrus.woff") format("woff"),
/* chrome firefox */ url("assets/fonts/papyrus.ttf") format("truetype"),
@ -270,13 +271,12 @@ ul {
padding: 0;
border: solid 0.125em transparent;
border-radius: 15px;
box-shadow: inset 0 1px 1px #9c9c9c, 0 0 10px #000;
background: radial-gradient(
at 0 -15%,
rgba(#fff, 0.07) 70%,
rgba(#fff, 0) 71%
)
0 0/ 80% 90% no-repeat content-box,
box-shadow:
inset 0 1px 1px #9c9c9c,
0 0 10px #000;
background:
radial-gradient(at 0 -15%, rgba(#fff, 0.07) 70%, rgba(#fff, 0) 71%) 0 0/ 80%
90% no-repeat content-box,
linear-gradient(#4e4e4e, #040404) content-box,
linear-gradient(#292929, #010101) border-box;
color: white;
@ -303,7 +303,8 @@ ul {
height: 10px;
}
&.townsfolk {
background: radial-gradient(
background:
radial-gradient(
at 0 -15%,
rgba(255, 255, 255, 0.07) 70%,
rgba(255, 255, 255, 0) 71%
@ -311,13 +312,16 @@ ul {
0 0/80% 90% no-repeat content-box,
linear-gradient(#0031ad, rgba(5, 0, 0, 0.22)) content-box,
linear-gradient(#292929, #001142) border-box;
box-shadow: inset 0 1px 1px #002c9c, 0 0 10px #000;
box-shadow:
inset 0 1px 1px #002c9c,
0 0 10px #000;
&:hover:not(.disabled) {
color: #008cf7;
}
}
&.demon {
background: radial-gradient(
background:
radial-gradient(
at 0 -15%,
rgba(255, 255, 255, 0.07) 70%,
rgba(255, 255, 255, 0) 71%
@ -325,7 +329,9 @@ ul {
0 0/80% 90% no-repeat content-box,
linear-gradient(#ad0000, rgba(5, 0, 0, 0.22)) content-box,
linear-gradient(#292929, #420000) border-box;
box-shadow: inset 0 1px 1px #9c0000, 0 0 10px #000;
box-shadow:
inset 0 1px 1px #9c0000,
0 0 10px #000;
}
}

View file

@ -2,7 +2,7 @@
{
"id": "_meta",
"logo": "https://cdn.shopify.com/oxygen/57467011226/524442/h11pcta59/build/_assets/Godfather_website_purple-3JVCH57E.png",
"name": "Late night drive by 1.6",
"name": "Late night drive by",
"author": "Aero"
},
{

View file

@ -6,13 +6,13 @@
export default {
props: {
timerName: String,
timerDuration: Number
timerDuration: Number,
},
computed: {
style() {
return `--timer: ${this.timerDuration}`;
}
}
},
},
};
</script>

View file

@ -29,10 +29,10 @@ export default {
["demon", "#ce0100", "#000"],
["townsfolk", "#1f65ff", "#000"],
["minion", "#ff6900", "#000"],
["default", "#4E4E4E", "#000"]
]
["default", "#4E4E4E", "#000"],
],
};
}
},
};
</script>

View file

@ -30,14 +30,14 @@ import { mapMutations, mapState, mapGetters } from "vuex";
export default {
computed: {
...mapState(["locale"]),
...mapGetters({ nightOrder: "players/nightOrder" })
...mapGetters({ nightOrder: "players/nightOrder" }),
},
data() {
return {
language: window.navigator.userLanguage || window.navigator.language
language: window.navigator.userLanguage || window.navigator.language,
};
},
methods: mapMutations(["toggleMenu"])
methods: mapMutations(["toggleMenu"]),
};
</script>

View file

@ -4,11 +4,9 @@
class="nomlog-summary"
v-show="session.voteHistory.length && session.sessionId"
@click="toggleModal('voteHistory')"
:title="
`${session.voteHistory.length} recent ${
session.voteHistory.length == 1 ? 'nomination' : 'nominations'
}`
"
:title="`${session.voteHistory.length} recent ${
session.voteHistory.length == 1 ? 'nomination' : 'nominations'
}`"
>
<font-awesome-icon icon="book-dead" />
{{ session.voteHistory.length }}
@ -17,15 +15,13 @@
class="session"
:class="{
spectator: session.isSpectator,
reconnecting: session.isReconnecting
reconnecting: session.isReconnecting,
}"
v-if="session.sessionId"
@click="leaveSession"
:title="
`${session.playerCount} other players in this session${
session.ping ? ' (' + session.ping + 'ms latency)' : ''
}`
"
:title="`${session.playerCount} other players in this session${
session.ping ? ' (' + session.ping + 'ms latency)' : ''
}`"
>
<font-awesome-icon icon="broadcast-tower" />
{{ session.playerCount }}
@ -76,7 +72,7 @@
<font-awesome-icon
:icon="[
'fas',
grimoire.isOrganVoteMode ? 'check-square' : 'square'
grimoire.isOrganVoteMode ? 'check-square' : 'square',
]"
/>
</em>
@ -87,7 +83,7 @@
<font-awesome-icon
:icon="[
'fas',
grimoire.isNightOrder ? 'check-square' : 'square'
grimoire.isNightOrder ? 'check-square' : 'square',
]"
/>
</em>
@ -108,7 +104,7 @@
</li>
<li @click="setBackground">
{{ locale.menu.grimoire.background }}
<em><font-awesome-icon icon="image"/></em>
<em><font-awesome-icon icon="image" /></em>
</li>
<li v-if="!edition.isOfficial" @click="imageOptIn">
<small>{{ locale.menu.grimoire.customImages }}</small>
@ -116,7 +112,7 @@
><font-awesome-icon
:icon="[
'fas',
grimoire.isImageOptIn ? 'check-square' : 'square'
grimoire.isImageOptIn ? 'check-square' : 'square',
]"
/></em>
</li>
@ -126,7 +122,7 @@
><font-awesome-icon
:icon="[
'fas',
grimoire.isStreamerMode ? 'check-square' : 'square'
grimoire.isStreamerMode ? 'check-square' : 'square',
]"
/></em>
</li>
@ -178,11 +174,11 @@
</li>
<li @click="copySessionUrl">
{{ locale.menu.session.link }}
<em><font-awesome-icon icon="copy"/></em>
<em><font-awesome-icon icon="copy" /></em>
</li>
<li v-if="!session.isSpectator" @click="distributeRoles">
{{ locale.menu.session.sendRoles }}
<em><font-awesome-icon icon="theater-masks"/></em>
<em><font-awesome-icon icon="theater-masks" /></em>
</li>
<li
v-if="session.voteHistory.length || !session.isSpectator"
@ -205,11 +201,11 @@
</li>
<li @click="randomizeSeatings" v-if="players.length > 2">
{{ locale.menu.players.randomize }}
<em><font-awesome-icon icon="dice"/></em>
<em><font-awesome-icon icon="dice" /></em>
</li>
<li @click="clearPlayers" v-if="players.length">
{{ locale.menu.players.removeAll }}
<em><font-awesome-icon icon="trash-alt"/></em>
<em><font-awesome-icon icon="trash-alt" /></em>
</li>
</template>
@ -229,11 +225,11 @@
</li>
<li v-if="!session.isSpectator" @click="toggleModal('fabled')">
{{ locale.menu.characters.addFabled }}
<em><font-awesome-icon icon="dragon"/></em>
<em><font-awesome-icon icon="dragon" /></em>
</li>
<li @click="clearRoles" v-if="players.length">
{{ locale.menu.characters.removeAll }}
<em><font-awesome-icon icon="trash-alt"/></em>
<em><font-awesome-icon icon="trash-alt" /></em>
</li>
</template>
@ -250,7 +246,7 @@
</li>
<li @click="toggleModal('gameState')">
{{ locale.menu.help.gameState }}
<em><font-awesome-icon icon="file-code"/></em>
<em><font-awesome-icon icon="file-code" /></em>
</li>
<li>
<a href="https://discord.gg/gD3AB8qCrw" target="_blank">
@ -284,11 +280,11 @@ import { mapMutations, mapState } from "vuex";
export default {
computed: {
...mapState(["grimoire", "session", "edition", "locale"]),
...mapState("players", ["players"])
...mapState("players", ["players"]),
},
data() {
return {
tab: "grimoire"
tab: "grimoire",
};
},
methods: {
@ -302,7 +298,7 @@ export default {
if (this.session.sessionId) return;
const sessionId = prompt(
this.locale.prompt.createSession,
Math.round(Math.random() * 10000)
Math.round(Math.random() * 10000),
);
if (sessionId) {
this.$store.commit("session/clearVoteHistory");
@ -326,7 +322,7 @@ export default {
(() => {
this.$store.commit("session/distributeRoles", false);
}).bind(this),
2000
2000,
);
}
},
@ -409,9 +405,9 @@ export default {
"toggleNightOrder",
"toggleStatic",
"setZoom",
"toggleModal"
])
}
"toggleModal",
]),
},
};
</script>

View file

@ -10,9 +10,9 @@
'no-vote': player.isVoteless,
you: session.sessionId && player.id && player.id === session.playerId,
'vote-yes': session.votes[index],
'vote-lock': voteLocked
'vote-lock': voteLocked,
},
player.role.team
player.role.team,
]"
>
<div class="shroud" @click="toggleStatus()"></div>
@ -22,7 +22,7 @@
class="night-order first"
v-if="
nightOrder.get(player).first &&
(grimoire.isNightOrder || !session.isSpectator)
(grimoire.isNightOrder || !session.isSpectator)
"
>
<em>{{ nightOrder.get(player).first }}.</em>
@ -34,7 +34,7 @@
class="night-order other"
v-if="
nightOrder.get(player).other &&
(grimoire.isNightOrder || !session.isSpectator)
(grimoire.isNightOrder || !session.isSpectator)
"
>
<em>{{ nightOrder.get(player).other }}.</em>
@ -53,8 +53,8 @@
<font-awesome-icon
v-if="
!grimoire.isOrganVoteMode ||
!session.isSpectator ||
player.id == session.playerId
!session.isSpectator ||
player.id == session.playerId
"
icon="hand-paper"
class="vote"
@ -64,8 +64,8 @@
<font-awesome-icon
v-if="
grimoire.isOrganVoteMode &&
session.isSpectator &&
player.id !== session.playerId
session.isSpectator &&
player.id !== session.playerId
"
icon="question"
class="vote"
@ -75,8 +75,8 @@
<font-awesome-icon
v-if="
!grimoire.isOrganVoteMode ||
!session.isSpectator ||
player.id == session.playerId
!session.isSpectator ||
player.id == session.playerId
"
icon="times"
class="vote"
@ -86,8 +86,8 @@
<font-awesome-icon
v-if="
grimoire.isOrganVoteMode &&
session.isSpectator &&
player.id !== session.playerId
session.isSpectator &&
player.id !== session.playerId
"
icon="question"
class="vote"
@ -159,7 +159,7 @@
@click="changePronouns"
v-if="
!session.isSpectator ||
(session.isSpectator && player.id === session.playerId)
(session.isSpectator && player.id === session.playerId)
"
>
<font-awesome-icon icon="venus-mars" />{{
@ -230,10 +230,12 @@
backgroundImage: `url(${
reminder.image && grimoire.isImageOptIn
? reminder.image
: require('../assets/icons/' +
(reminder.imageAlt || reminder.role) +
'.png')
})`
: require(
'../assets/icons/' +
(reminder.imageAlt || reminder.role) +
'.png',
)
})`,
}"
></span>
<span class="text">{{ reminder.name }}</span>
@ -252,22 +254,22 @@ import { mapGetters, mapState } from "vuex";
export default {
components: {
Token
Token,
},
props: {
player: {
type: Object,
required: true
}
required: true,
},
},
computed: {
...mapState("players", ["players"]),
...mapState(["grimoire", "session", "locale"]),
...mapGetters({ nightOrder: "players/nightOrder" }),
index: function() {
index: function () {
return this.players.indexOf(this.player);
},
voteLocked: function() {
voteLocked: function () {
const session = this.session;
const players = this.players.length;
if (!session.nomination) return false;
@ -275,7 +277,7 @@ export default {
(this.index - 1 + players - session.nomination[1]) % players;
return indexAdjusted < session.lockedVote - 1;
},
zoom: function() {
zoom: function () {
if (this.players.length < 7) {
return { width: 18 + this.grimoire.zoom + "vmin" };
} else if (this.players.length <= 10) {
@ -285,12 +287,12 @@ export default {
} else {
return { width: 12 + this.grimoire.zoom + "vmin" };
}
}
},
},
data() {
return {
isMenuOpen: false,
isSwap: false
isSwap: false,
};
},
methods: {
@ -346,7 +348,7 @@ export default {
this.$store.commit("players/update", {
player: this.player,
property,
value
value,
});
if (closeMenu) {
this.isMenuOpen = false;
@ -383,10 +385,10 @@ export default {
if (!this.voteLocked) return;
this.$store.commit("session/voteSync", [
this.index,
!this.session.votes[this.index]
!this.session.votes[this.index],
]);
}
}
},
},
};
</script>
@ -924,7 +926,10 @@ li.move:not(.from) .player .overlay svg.move {
width: 100%;
position: absolute;
top: 15%;
text-shadow: 0 1px 1px #f6dfbd, 0 -1px 1px #f6dfbd, 1px 0 1px #f6dfbd,
text-shadow:
0 1px 1px #f6dfbd,
0 -1px 1px #f6dfbd,
1px 0 1px #f6dfbd,
-1px 0 1px #f6dfbd;
}

View file

@ -8,7 +8,7 @@
role.image && grimoire.isImageOptIn
? role.image
: require('../assets/icons/' + (role.imageAlt || role.id) + '.png')
})`
})`,
}"
></span>
<span
@ -54,29 +54,29 @@ export default {
props: {
role: {
type: Object,
default: () => ({})
}
default: () => ({}),
},
},
computed: {
reminderLeaves: function() {
reminderLeaves: function () {
return (
(this.role.reminders || []).length +
(this.role.remindersGlobal || []).length
);
},
...mapState(["grimoire"])
...mapState(["grimoire"]),
},
data() {
return {};
},
filters: {
nameToFontSize: name => (name && name.length > 10 ? "90%" : "110%")
nameToFontSize: (name) => (name && name.length > 10 ? "90%" : "110%"),
},
methods: {
setRole() {
this.$emit("set-role");
}
}
},
},
};
</script>

View file

@ -8,7 +8,7 @@
edition.logo && grimoire.isImageOptIn
? edition.logo
: require('../assets/editions/' + edition.id + '.png')
})`
})`,
}"
></li>
<li v-if="players.length - teams.traveler < 5">
@ -99,15 +99,15 @@ import Countdown from "./Countdown";
export default {
components: {
Countdown
Countdown,
},
computed: {
teams: function() {
teams: function () {
const { players } = this.$store.state.players;
const nonTravelers = this.$store.getters["players/nonTravelers"];
const alive = players.filter(player => player.isDead !== true).length;
const alive = players.filter((player) => player.isDead !== true).length;
const aliveNT = players.filter(
player => player.isDead !== true && player.role.team !== "traveler"
(player) => player.isDead !== true && player.role.team !== "traveler",
).length;
return {
...gameJSON[nonTravelers - 5],
@ -117,16 +117,16 @@ export default {
votes:
alive +
players.filter(
player => player.isDead === true && player.isVoteless !== true
).length
(player) => player.isDead === true && player.isVoteless !== true,
).length,
};
},
countdownStyle: function() {
countdownStyle: function () {
return `--timer: ${this.$store.state.grimoire.timer.duration}`;
},
...mapState(["edition", "grimoire", "locale"]),
...mapState("players", ["players"])
}
...mapState("players", ["players"]),
},
};
</script>
@ -153,7 +153,10 @@ export default {
display: flex;
flex-wrap: wrap;
justify-content: center;
text-shadow: 0 2px 1px black, 0 -2px 1px black, 2px 0 1px black,
text-shadow:
0 2px 1px black,
0 -2px 1px black,
2px 0 1px black,
-2px 0 1px black;
span {

View file

@ -5,7 +5,7 @@
:class="{
public: grimoire.isPublic,
spectator: session.isSpectator,
vote: session.nomination
vote: session.nomination,
}"
>
<ul class="circle" :class="['size-' + players.length]">
@ -18,7 +18,7 @@
from: Math.max(swap, move, nominate) === index,
swap: swap > -1,
move: move > -1,
nominate: nominate > -1
nominate: nominate > -1,
}"
></Player>
</ul>
@ -142,7 +142,7 @@
class="night-order first"
v-if="
nightOrder.get(role).first &&
(grimoire.isNightOrder || !session.isSpectator)
(grimoire.isNightOrder || !session.isSpectator)
"
>
<em>{{ nightOrder.get(role).first }}.</em>
@ -154,7 +154,7 @@
class="night-order other"
v-if="
nightOrder.get(role).other &&
(grimoire.isNightOrder || !session.isSpectator)
(grimoire.isNightOrder || !session.isSpectator)
"
>
<em>{{ nightOrder.get(role).other }}.</em>
@ -184,12 +184,12 @@ export default {
Player,
Token,
RoleModal,
ReminderModal
ReminderModal,
},
computed: {
...mapGetters({ nightOrder: "players/nightOrder" }),
...mapState(["grimoire", "roles", "session", "locale"]),
...mapState("players", ["players", "bluffs", "fabled"])
...mapState("players", ["players", "bluffs", "fabled"]),
},
data() {
return {
@ -204,7 +204,7 @@ export default {
timerName: "Timer",
timerDuration: 1,
timerOn: false,
timerEnder: false
timerEnder: false,
};
},
methods: {
@ -259,7 +259,7 @@ export default {
if (this.session.isSpectator || this.session.lockedVote) return;
if (
confirm(
`Do you really want to remove ${this.players[playerIndex].name}?`
`Do you really want to remove ${this.players[playerIndex].name}?`,
)
) {
const { nomination } = this.session;
@ -274,7 +274,7 @@ export default {
// update nomination array if removed player has lower index
this.$store.commit("session/setNomination", [
nomination[0] > playerIndex ? nomination[0] - 1 : nomination[0],
nomination[1] > playerIndex ? nomination[1] - 1 : nomination[1]
nomination[1] > playerIndex ? nomination[1] - 1 : nomination[1],
]);
}
}
@ -290,7 +290,7 @@ export default {
if (this.session.nomination) {
// update nomination if one of the involved players is swapped
const swapTo = this.players.indexOf(to);
const updatedNomination = this.session.nomination.map(nom => {
const updatedNomination = this.session.nomination.map((nom) => {
if (nom === this.swap) return swapTo;
if (nom === swapTo) return this.swap;
return nom;
@ -304,7 +304,7 @@ export default {
}
this.$store.commit("players/swap", [
this.swap,
this.players.indexOf(to)
this.players.indexOf(to),
]);
this.cancel();
}
@ -318,7 +318,7 @@ export default {
if (this.session.nomination) {
// update nomination if it is affected by the move
const moveTo = this.players.indexOf(to);
const updatedNomination = this.session.nomination.map(nom => {
const updatedNomination = this.session.nomination.map((nom) => {
if (nom === this.move) return moveTo;
if (nom > this.move && nom <= moveTo) return nom - 1;
if (nom < this.move && nom >= moveTo) return nom + 1;
@ -333,7 +333,7 @@ export default {
}
this.$store.commit("players/move", [
this.move,
this.players.indexOf(to)
this.players.indexOf(to),
]);
this.cancel();
}
@ -359,7 +359,7 @@ export default {
renameTimer() {
let newName = prompt(
this.locale.townsquare.timer.prompt.name,
this.timerName
this.timerName,
);
if (newName === "") {
newName = this.locale.townsquare.timer.default.text;
@ -372,7 +372,8 @@ export default {
},
setNominationTimer() {
this.timerDuration = 2;
this.timerName = this.timerName = this.locale.townsquare.timer.nominations.text;
this.timerName = this.timerName =
this.locale.townsquare.timer.nominations.text;
},
setDuskTimer() {
this.timerDuration = 1;
@ -399,7 +400,7 @@ export default {
let timerText = this.locale.townsquare.timer.debate.text;
timerText = timerText.replace(
"$accusee",
this.players[this.session.nomination[1]].name
this.players[this.session.nomination[1]].name,
);
this.timerName = timerText;
},
@ -422,8 +423,8 @@ export default {
this.$store.commit("setTimer", {});
this.timerOn = false;
clearTimeout(this.timerEnder);
}
}
},
},
};
</script>

View file

@ -6,16 +6,21 @@
</div>
<div class="overlay">
<audio src="../assets/sounds/countdown.mp3" preload="auto"></audio>
<em class="blue">{{ nominator.name }}</em> {{ locale.vote.nominated }}
<em class="blue">{{ nominator.name }}</em>
{{
nominee.role.team == "traveler"
? locale.vote.callexile
: locale.vote.nominates
}}
<em>{{ nominee.name }}</em
>!
>{{ locale.vote.exclam }}
<br />
<em
class="blue"
v-if="
!grimoire.isOrganVoteMode ||
nominee.role.team == 'traveler' ||
!session.isSpectator
nominee.role.team == 'traveler' ||
!session.isSpectator
"
>
{{ voters.length }} {{ locale.vote.votes }}
@ -71,7 +76,7 @@
<div
class="button"
:class="{
disabled: session.nomination[1] === session.markedPlayer
disabled: session.nomination[1] === session.markedPlayer,
}"
@click="setMarked"
>
@ -137,44 +142,46 @@ import Countdown from "./Countdown";
export default {
components: {
Countdown
Countdown,
},
computed: {
...mapState("players", ["players"]),
...mapState(["session", "grimoire", "locale"]),
...mapGetters({ alive: "players/alive" }),
nominator: function() {
nominator: function () {
return this.players[this.session.nomination[0]];
},
nominatorStyle: function() {
nominatorStyle: function () {
const players = this.players.length;
const nomination = this.session.nomination[0];
return {
transform: `rotate(${Math.round((nomination / players) * 360)}deg)`,
transitionDuration: this.session.votingSpeed - 100 + "ms"
transitionDuration: this.session.votingSpeed - 100 + "ms",
};
},
nominee: function() {
nominee: function () {
return this.players[this.session.nomination[1]];
},
nomineeStyle: function() {
nomineeStyle: function () {
const players = this.players.length;
const nomination = this.session.nomination[1];
const lock = this.session.lockedVote;
const rotation = (360 * (nomination + Math.min(lock, players))) / players;
return {
transform: `rotate(${Math.round(rotation)}deg)`,
transitionDuration: this.session.votingSpeed - 100 + "ms"
transitionDuration: this.session.votingSpeed - 100 + "ms",
};
},
player: function() {
return this.players.find(p => p.id === this.session.playerId);
player: function () {
return this.players.find((p) => p.id === this.session.playerId);
},
currentVote: function() {
const index = this.players.findIndex(p => p.id === this.session.playerId);
currentVote: function () {
const index = this.players.findIndex(
(p) => p.id === this.session.playerId,
);
return index >= 0 ? !!this.session.votes[index] : undefined;
},
canVote: function() {
canVote: function () {
if (!this.player) return false;
if (this.player.isVoteless && this.nominee.role.team !== "traveler")
return false;
@ -185,26 +192,27 @@ export default {
(index - 1 + players - session.nomination[1]) % players;
return indexAdjusted >= session.lockedVote - 1;
},
voters: function() {
voters: function () {
const nomination = this.session.nomination[1];
const voters = Array(this.players.length)
.fill("")
.map((x, index) =>
this.session.votes[index] ? this.players[index].name : ""
this.session.votes[index] ? this.players[index].name : "",
);
const reorder = [
...voters.slice(nomination + 1),
...voters.slice(0, nomination + 1)
...voters.slice(0, nomination + 1),
];
return (this.session.lockedVote
? reorder.slice(0, this.session.lockedVote - 1)
: reorder
).filter(n => !!n);
}
return (
this.session.lockedVote
? reorder.slice(0, this.session.lockedVote - 1)
: reorder
).filter((n) => !!n);
},
},
data() {
return {
voteTimer: null
voteTimer: null,
};
},
methods: {
@ -254,7 +262,9 @@ export default {
},
vote(vote) {
if (!this.canVote) return false;
const index = this.players.findIndex(p => p.id === this.session.playerId);
const index = this.players.findIndex(
(p) => p.id === this.session.playerId,
);
if (index >= 0 && !!this.session.votes[index] !== vote) {
this.$store.commit("session/voteSync", [index, vote]);
}
@ -270,8 +280,8 @@ export default {
},
removeMarked() {
this.$store.commit("session/setMarkedPlayer", -1);
}
}
},
},
};
</script>
@ -289,7 +299,10 @@ export default {
background: url("../assets/demon-head.png") center center no-repeat;
background-size: auto 75%;
text-align: center;
text-shadow: 0 1px 2px #000000, 0 -1px 2px #000000, 1px 0 2px #000000,
text-shadow:
0 1px 2px #000000,
0 -1px 2px #000000,
1px 0 2px #000000,
-1px 0 2px #000000;
.mark .button {

View file

@ -39,9 +39,9 @@
class="edition"
:class="['edition-' + edition.id]"
:style="{
backgroundImage: `url(${require('../../assets/editions/' +
edition.id +
'.png')})`
backgroundImage: `url(${require(
'../../assets/editions/' + edition.id + '.png',
)})`,
}"
:key="edition.id"
@click="setEdition(edition)"
@ -120,15 +120,15 @@ import Modal from "./Modal";
export default {
components: {
Modal
Modal,
},
data: function() {
data: function () {
return {
tab: "official"
tab: "official",
};
},
computed: {
...mapState(["modals", "locale", "editions"])
...mapState(["modals", "locale", "editions"]),
},
methods: {
openUpload() {
@ -178,7 +178,9 @@ export default {
},
parseRoles(roles) {
if (!roles || !roles.length) return;
roles = roles.map(role => typeof role === "string" ? { id: role } : role);
roles = roles.map((role) =>
typeof role === "string" ? { id: role } : role,
);
const metaIndex = roles.findIndex(({ id }) => id === "_meta");
let meta = {};
if (metaIndex > -1) {
@ -187,7 +189,7 @@ export default {
this.$store.commit("setCustomRoles", roles);
this.$store.commit(
"setEdition",
Object.assign({}, meta, { id: "custom" })
Object.assign({}, meta, { id: "custom" }),
);
// check for fabled and set those too, if present
if (roles.some((role) => this.$store.state.fabled.has(role.id || role))) {
@ -200,8 +202,8 @@ export default {
this.$store.commit("players/setFabled", { fabled });
}
},
...mapMutations(["toggleModal", "setEdition"])
}
...mapMutations(["toggleModal", "setEdition"]),
},
};
</script>
@ -222,8 +224,12 @@ ul.editions {
width: 30%;
margin: 5px;
font-size: 120%;
text-shadow: -1px -1px 0 #000, 1px -1px 0 #000, -1px 1px 0 #000,
1px 1px 0 #000, 0 0 5px rgba(0, 0, 0, 0.75);
text-shadow:
-1px -1px 0 #000,
1px -1px 0 #000,
-1px 1px 0 #000,
1px 1px 0 #000,
0 0 5px rgba(0, 0, 0, 0.75);
cursor: pointer;
&:hover {
color: red;

View file

@ -20,26 +20,28 @@ export default {
...mapState(["modals", "fabled", "grimoire", "locale"]),
fabled() {
const fabled = [];
this.$store.state.fabled.forEach(role => {
this.$store.state.fabled.forEach((role) => {
// don't show fabled that are already in play
if (
!this.$store.state.players.fabled.some(fable => fable.id === role.id)
!this.$store.state.players.fabled.some(
(fable) => fable.id === role.id,
)
) {
fabled.push(role);
}
});
return fabled;
}
},
},
methods: {
setFabled(role) {
this.$store.commit("players/setFabled", {
fabled: role
fabled: role,
});
this.$store.commit("toggleModal", "fabled");
},
...mapMutations(["toggleModal"])
}
...mapMutations(["toggleModal"]),
},
};
</script>

View file

@ -28,10 +28,10 @@ import { mapMutations, mapState } from "vuex";
export default {
components: {
Modal
Modal,
},
computed: {
gamestate: function() {
gamestate: function () {
return JSON.stringify({
bluffs: this.players.bluffs.map(({ id }) => id),
edition: this.edition.isOfficial
@ -40,27 +40,27 @@ export default {
roles: this.edition.isOfficial
? ""
: this.$store.getters.customRolesStripped,
fabled: this.players.fabled.map(fabled =>
fabled.isCustom ? fabled : { id: fabled.id }
fabled: this.players.fabled.map((fabled) =>
fabled.isCustom ? fabled : { id: fabled.id },
),
players: this.players.players.map(player => ({
players: this.players.players.map((player) => ({
...player,
role: player.role.id || {}
}))
role: player.role.id || {},
})),
});
},
...mapState(["modals", "players", "edition", "roles", "session", "locale"])
...mapState(["modals", "players", "edition", "roles", "session", "locale"]),
},
data() {
return {
input: ""
input: "",
};
},
methods: {
copy: function() {
copy: function () {
navigator.clipboard.writeText(this.input || this.gamestate);
},
load: function() {
load: function () {
if (this.session.isSpectator) return;
try {
const data = JSON.parse(this.input || this.gamestate);
@ -75,30 +75,30 @@ export default {
bluffs.forEach((role, index) => {
this.$store.commit("players/setBluff", {
index,
role: this.$store.state.roles.get(role) || {}
role: this.$store.state.roles.get(role) || {},
});
});
}
if (fabled) {
this.$store.commit("players/setFabled", {
fabled: fabled.map(
f =>
(f) =>
this.$store.state.fabled.get(f) ||
this.$store.state.fabled.get(f.id) ||
f
)
f,
),
});
}
if (players) {
this.$store.commit(
"players/set",
players.map(player => ({
players.map((player) => ({
...player,
role:
this.$store.state.roles.get(player.role) ||
this.$store.getters.rolesJSONbyId.get(player.role) ||
{}
}))
{},
})),
);
}
this.toggleModal("gameState");
@ -106,8 +106,8 @@ export default {
alert("Unable to parse JSON: " + e);
}
},
...mapMutations(["toggleModal"])
}
...mapMutations(["toggleModal"]),
},
};
</script>

View file

@ -31,16 +31,16 @@
<script>
export default {
data: function() {
data: function () {
return {
isMaximized: false
isMaximized: false,
};
},
methods: {
close() {
this.$emit("close");
}
}
},
},
};
</script>

View file

@ -44,10 +44,12 @@
backgroundImage: `url(${
role.image && grimoire.isImageOptIn
? role.image
: require('../../assets/icons/' +
(role.imageAlt || role.id) +
'.png')
})`
: require(
'../../assets/icons/' +
(role.imageAlt || role.id) +
'.png',
)
})`,
}"
></span>
<span class="reminder" v-if="role.firstNightReminder">
@ -69,10 +71,12 @@
backgroundImage: `url(${
role.image && grimoire.isImageOptIn
? role.image
: require('../../assets/icons/' +
(role.imageAlt || role.id) +
'.png')
})`
: require(
'../../assets/icons/' +
(role.imageAlt || role.id) +
'.png',
)
})`,
}"
></span>
<span class="name">
@ -104,10 +108,10 @@ import { mapMutations, mapState } from "vuex";
export default {
components: {
Modal
Modal,
},
computed: {
rolesFirstNight: function() {
rolesFirstNight: function () {
const rolesFirstNight = [];
// Ajouter minion / demon infos à l'ordre nocturne
if (this.players.length > 6) {
@ -117,57 +121,57 @@ export default {
name: this.locale.modal.nightOrder.minionInfo,
firstNight: 5,
team: "minion",
players: this.players.filter(p => p.role.team === "minion"),
firstNightReminder: this.locale.modal.nightOrder
.minionInfoDescription
players: this.players.filter((p) => p.role.team === "minion"),
firstNightReminder:
this.locale.modal.nightOrder.minionInfoDescription,
},
{
id: "evil",
name: this.locale.modal.nightOrder.demonInfo,
firstNight: 8,
team: "demon",
players: this.players.filter(p => p.role.team === "demon"),
firstNightReminder: this.locale.modal.nightOrder
.demonInfoDescription
}
players: this.players.filter((p) => p.role.team === "demon"),
firstNightReminder:
this.locale.modal.nightOrder.demonInfoDescription,
},
);
}
this.roles.forEach(role => {
const players = this.players.filter(p => p.role.id === role.id);
this.roles.forEach((role) => {
const players = this.players.filter((p) => p.role.id === role.id);
if (role.firstNight && (role.team !== "traveler" || players.length)) {
rolesFirstNight.push(Object.assign({ players }, role));
}
});
this.fabled
.filter(({ firstNight }) => firstNight)
.forEach(fabled => {
.forEach((fabled) => {
rolesFirstNight.push(Object.assign({ players: [] }, fabled));
});
rolesFirstNight.sort((a, b) => a.firstNight - b.firstNight);
return rolesFirstNight;
},
rolesOtherNight: function() {
rolesOtherNight: function () {
const rolesOtherNight = [];
this.roles.forEach(role => {
const players = this.players.filter(p => p.role.id === role.id);
this.roles.forEach((role) => {
const players = this.players.filter((p) => p.role.id === role.id);
if (role.otherNight && (role.team !== "traveler" || players.length)) {
rolesOtherNight.push(Object.assign({ players }, role));
}
});
this.fabled
.filter(({ otherNight }) => otherNight)
.forEach(fabled => {
.forEach((fabled) => {
rolesOtherNight.push(Object.assign({ players: [] }, fabled));
});
rolesOtherNight.sort((a, b) => a.otherNight - b.otherNight);
return rolesOtherNight;
},
...mapState(["roles", "modals", "edition", "grimoire", "locale"]),
...mapState("players", ["players", "fabled"])
...mapState("players", ["players", "fabled"]),
},
methods: {
...mapMutations(["toggleModal"])
}
...mapMutations(["toggleModal"]),
},
};
</script>

View file

@ -32,10 +32,12 @@
backgroundImage: `url(${
role.image && grimoire.isImageOptIn
? role.image
: require('../../assets/icons/' +
(role.imageAlt || role.id) +
'.png')
})`
: require(
'../../assets/icons/' +
(role.imageAlt || role.id) +
'.png',
)
})`,
}"
></span>
<div class="role">
@ -60,17 +62,17 @@
<span
class="icon"
:style="{
backgroundImage: `url(${require('../../assets/icons/' +
jinx.first.id +
'.png')})`
backgroundImage: `url(${require(
'../../assets/icons/' + jinx.first.id + '.png',
)})`,
}"
></span>
<span
class="icon"
:style="{
backgroundImage: `url(${require('../../assets/icons/' +
jinx.second.id +
'.png')})`
backgroundImage: `url(${require(
'../../assets/icons/' + jinx.second.id + '.png',
)})`,
}"
></span>
<div class="role">
@ -84,6 +86,7 @@
<li></li>
</ul>
</div>
<div class="asterisk">{{ locale.modal.reference.notfirstnight }}</div>
</Modal>
</template>
@ -93,23 +96,23 @@ import { mapMutations, mapState } from "vuex";
export default {
components: {
Modal
Modal,
},
computed: {
/**
* Return a list of jinxes in the form of role IDs and a reason
* @returns {*[]} [{first, second, reason}]
*/
jinxed: function() {
jinxed: function () {
const jinxed = [];
this.roles.forEach(role => {
this.roles.forEach((role) => {
if (this.jinxes.get(role.id)) {
this.jinxes.get(role.id).forEach((reason, second) => {
if (this.roles.get(second)) {
jinxed.push({
first: role,
second: this.roles.get(second),
reason
reason,
});
}
});
@ -117,9 +120,9 @@ export default {
});
return jinxed;
},
rolesGrouped: function() {
rolesGrouped: function () {
const rolesGrouped = {};
this.roles.forEach(role => {
this.roles.forEach((role) => {
if (!rolesGrouped[role.team]) {
rolesGrouped[role.team] = [];
}
@ -128,7 +131,7 @@ export default {
delete rolesGrouped["traveler"];
return rolesGrouped;
},
playersByRole: function() {
playersByRole: function () {
const players = {};
this.players.forEach(({ name, role }) => {
if (role && role.id && role.team !== "traveler") {
@ -141,11 +144,11 @@ export default {
return players;
},
...mapState(["roles", "modals", "edition", "grimoire", "jinxes", "locale"]),
...mapState("players", ["players"])
...mapState("players", ["players"]),
},
methods: {
...mapMutations(["toggleModal"])
}
...mapMutations(["toggleModal"]),
},
};
</script>
@ -211,6 +214,12 @@ h3 {
}
}
.asterisk {
font-size: 60%;
text-align: right;
padding-top: 20px;
}
.team {
display: flex;
align-items: stretch;

View file

@ -18,10 +18,12 @@
backgroundImage: `url(${
reminder.image && grimoire.isImageOptIn
? reminder.image
: require('../../assets/icons/' +
(reminder.imageAlt || reminder.role) +
'.png')
})`
: require(
'../../assets/icons/' +
(reminder.imageAlt || reminder.role) +
'.png',
)
})`,
}"
></span>
<span class="text">{{ reminder.name }}</span>
@ -39,12 +41,14 @@ import { mapMutations, mapState } from "vuex";
* @param role The role for which the reminder should be generated
* @return {function(*): {image: string|string[]|string|*, role: *, name: *, imageAlt: string|*}}
*/
const mapReminder = ({ id, image, imageAlt }) => name => ({
role: id,
image,
imageAlt,
name
});
const mapReminder =
({ id, image, imageAlt }) =>
(name) => ({
role: id,
image,
imageAlt,
name,
});
export default {
components: { Modal },
@ -53,31 +57,31 @@ export default {
availableReminders() {
let reminders = [];
const { players, bluffs } = this.$store.state.players;
this.$store.state.roles.forEach(role => {
this.$store.state.roles.forEach((role) => {
// add reminders from player roles
if (players.some(p => p.role.id === role.id)) {
if (players.some((p) => p.role.id === role.id)) {
reminders = [...reminders, ...role.reminders.map(mapReminder(role))];
}
// add reminders from bluff/other roles
else if (bluffs.some(bluff => bluff.id === role.id)) {
else if (bluffs.some((bluff) => bluff.id === role.id)) {
reminders = [...reminders, ...role.reminders.map(mapReminder(role))];
}
// add global reminders
if (role.remindersGlobal && role.remindersGlobal.length) {
reminders = [
...reminders,
...role.remindersGlobal.map(mapReminder(role))
...role.remindersGlobal.map(mapReminder(role)),
];
}
});
// add fabled reminders
this.$store.state.players.fabled.forEach(role => {
this.$store.state.players.fabled.forEach((role) => {
reminders = [...reminders, ...role.reminders.map(mapReminder(role))];
});
// add out of script traveler reminders
this.$store.state.otherTravelers.forEach(role => {
if (players.some(p => p.role.id === role.id)) {
this.$store.state.otherTravelers.forEach((role) => {
if (players.some((p) => p.role.id === role.id)) {
reminders = [...reminders, ...role.reminders.map(mapReminder(role))];
}
});
@ -86,12 +90,12 @@ export default {
reminders.push({ role: "evil", name: this.locale.modal.reminder.evil });
reminders.push({
role: "custom",
name: this.locale.modal.reminder.custom
name: this.locale.modal.reminder.custom,
});
return reminders;
},
...mapState(["modals", "grimoire", "locale"]),
...mapState("players", ["players"])
...mapState("players", ["players"]),
},
methods: {
addReminder(reminder) {
@ -107,12 +111,12 @@ export default {
this.$store.commit("players/update", {
player,
property: "reminders",
value
value,
});
this.$store.commit("toggleModal", "reminder");
},
...mapMutations(["toggleModal"])
}
...mapMutations(["toggleModal"]),
},
};
</script>

View file

@ -60,12 +60,12 @@ export default {
availableRoles() {
const availableRoles = [];
const players = this.$store.state.players.players;
this.$store.state.roles.forEach(role => {
this.$store.state.roles.forEach((role) => {
// don't show bluff roles that are already assigned to players
if (
this.playerIndex >= 0 ||
(this.playerIndex < 0 &&
!players.some(player => player.role.id === role.id))
!players.some((player) => player.role.id === role.id))
) {
availableRoles.push(role);
}
@ -75,11 +75,11 @@ export default {
},
...mapState(["modals", "roles", "session", "locale"]),
...mapState("players", ["players"]),
...mapState(["otherTravelers"])
...mapState(["otherTravelers"]),
},
data() {
return {
tab: "editionRoles"
tab: "editionRoles",
};
},
methods: {
@ -88,7 +88,7 @@ export default {
// assign to bluff slot (index < 0)
this.$store.commit("players/setBluff", {
index: this.playerIndex * -1 - 1,
role
role,
});
} else {
if (this.session.isSpectator && role.team === "traveler") return;
@ -97,7 +97,7 @@ export default {
this.$store.commit("players/update", {
player,
property: "role",
value: role
value: role,
});
}
this.tab = "editionRoles";
@ -107,8 +107,8 @@ export default {
this.tab = "editionRoles";
this.toggleModal("role");
},
...mapMutations(["toggleModal"])
}
...mapMutations(["toggleModal"]),
},
};
</script>
@ -122,19 +122,29 @@ ul.tokens li {
transition: transform 500ms ease;
&.townsfolk {
box-shadow: 0 0 10px $townsfolk, 0 0 10px #004cff;
box-shadow:
0 0 10px $townsfolk,
0 0 10px #004cff;
}
&.outsider {
box-shadow: 0 0 10px $outsider, 0 0 10px $outsider;
box-shadow:
0 0 10px $outsider,
0 0 10px $outsider;
}
&.minion {
box-shadow: 0 0 10px $minion, 0 0 10px $minion;
box-shadow:
0 0 10px $minion,
0 0 10px $minion;
}
&.demon {
box-shadow: 0 0 10px $demon, 0 0 10px $demon;
box-shadow:
0 0 10px $demon,
0 0 10px $demon;
}
&.traveler {
box-shadow: 0 0 10px $traveler, 0 0 10px $traveler;
box-shadow:
0 0 10px $traveler,
0 0 10px $traveler;
}
&:hover {
transform: scale(1.2);

View file

@ -7,8 +7,8 @@
<h3>
{{
locale.modal.roles.titleStart +
nonTravelers +
locale.modal.roles.titleEnd
nonTravelers +
locale.modal.roles.titleEnd
}}
</h3>
<ul class="tokens" v-for="(teamRoles, team) in roleSelection" :key="team">
@ -48,14 +48,14 @@
class="button"
@click="assignRoles"
:class="{
disabled: selectedRoles > nonTravelers || !selectedRoles
disabled: selectedRoles > nonTravelers || !selectedRoles,
}"
>
<font-awesome-icon icon="people-arrows" />
{{
locale.modal.roles.assignStart +
selectedRoles +
locale.modal.roles.assignEnd
selectedRoles +
locale.modal.roles.assignEnd
}}
</div>
<div class="button" @click="selectRandomRoles">
@ -72,39 +72,39 @@ import gameJSON from "./../../game";
import Token from "./../Token";
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)];
export default {
components: {
Token,
Modal
Modal,
},
data: function() {
data: function () {
return {
roleSelection: {},
game: gameJSON,
allowMultiple: false
allowMultiple: false,
};
},
computed: {
selectedRoles: function() {
selectedRoles: function () {
return Object.values(this.roleSelection)
.map(roles => roles.reduce((a, { selected }) => a + selected, 0))
.map((roles) => roles.reduce((a, { selected }) => a + selected, 0))
.reduce((a, b) => a + b, 0);
},
hasSelectedSetupRoles: function() {
return Object.values(this.roleSelection).some(roles =>
roles.some(role => role.selected && role.setup)
hasSelectedSetupRoles: function () {
return Object.values(this.roleSelection).some((roles) =>
roles.some((role) => role.selected && role.setup),
);
},
...mapState(["roles", "modals", "locale"]),
...mapState("players", ["players"]),
...mapGetters({ nonTravelers: "players/nonTravelers" })
...mapGetters({ nonTravelers: "players/nonTravelers" }),
},
methods: {
selectRandomRoles() {
this.roleSelection = {};
this.roles.forEach(role => {
this.roles.forEach((role) => {
if (!this.roleSelection[role.team]) {
this.$set(this.roleSelection, role.team, []);
}
@ -114,11 +114,11 @@ export default {
delete this.roleSelection["traveler"];
const playerCount = Math.max(5, this.nonTravelers);
const composition = this.game[playerCount - 5];
Object.keys(composition).forEach(team => {
Object.keys(composition).forEach((team) => {
for (let x = 0; x < composition[team]; x++) {
if (this.roleSelection[team]) {
const available = this.roleSelection[team].filter(
role => !role.selected
(role) => !role.selected,
);
if (available.length) {
randomElement(available).selected = 1;
@ -131,32 +131,32 @@ export default {
if (this.selectedRoles <= this.nonTravelers && this.selectedRoles) {
// generate list of selected roles and randomize it
const roles = Object.values(this.roleSelection)
.map(roles =>
.map((roles) =>
roles
// duplicate roles selected more than once and filter unselected
.reduce((a, r) => [...a, ...Array(r.selected).fill(r)], [])
.reduce((a, r) => [...a, ...Array(r.selected).fill(r)], []),
)
// flatten into a single array
.reduce((a, b) => [...a, ...b], [])
.map(a => [Math.random(), a])
.map((a) => [Math.random(), a])
.sort((a, b) => a[0] - b[0])
.map(a => a[1]);
this.players.forEach(player => {
.map((a) => a[1]);
this.players.forEach((player) => {
if (player.role.team !== "traveler" && roles.length) {
const value = roles.pop();
this.$store.commit("players/update", {
player,
property: "role",
value
value,
});
}
});
this.$store.commit("toggleModal", "roles");
}
},
...mapMutations(["toggleModal"])
...mapMutations(["toggleModal"]),
},
mounted: function() {
mounted: function () {
if (!Object.keys(this.roleSelection).length) {
this.selectRandomRoles();
}
@ -164,8 +164,8 @@ export default {
watch: {
roles() {
this.selectRandomRoles();
}
}
},
},
};
</script>
@ -190,19 +190,29 @@ ul.tokens {
}
}
&.townsfolk {
box-shadow: 0 0 10px $townsfolk, 0 0 10px #004cff;
box-shadow:
0 0 10px $townsfolk,
0 0 10px #004cff;
}
&.outsider {
box-shadow: 0 0 10px $outsider, 0 0 10px $outsider;
box-shadow:
0 0 10px $outsider,
0 0 10px $outsider;
}
&.minion {
box-shadow: 0 0 10px $minion, 0 0 10px $minion;
box-shadow:
0 0 10px $minion,
0 0 10px $minion;
}
&.demon {
box-shadow: 0 0 10px $demon, 0 0 10px $demon;
box-shadow:
0 0 10px $demon,
0 0 10px $demon;
}
&.traveler {
box-shadow: 0 0 10px $traveler, 0 0 10px $traveler;
box-shadow:
0 0 10px $traveler,
0 0 10px $traveler;
}
&:hover {
transform: scale(1.2);

View file

@ -20,7 +20,7 @@
<font-awesome-icon
:icon="[
'fas',
session.isVoteHistoryAllowed ? 'check-square' : 'square'
session.isVoteHistoryAllowed ? 'check-square' : 'square',
]"
/>
{{ locale.modal.voteHistory.accessibility }}
@ -49,16 +49,8 @@
<tbody>
<tr v-for="(vote, index) in session.voteHistory" :key="index">
<td>
{{
vote.timestamp
.getHours()
.toString()
.padStart(2, "0")
}}:{{
vote.timestamp
.getMinutes()
.toString()
.padStart(2, "0")
{{ vote.timestamp.getHours().toString().padStart(2, "0") }}:{{
vote.timestamp.getMinutes().toString().padStart(2, "0")
}}
</td>
<td>{{ vote.nominator }}</td>
@ -77,7 +69,7 @@
? 'minus-square'
: vote.votes.length >= vote.majority
? 'check-square'
: 'square'
: 'square',
]"
/>
</td>
@ -100,10 +92,10 @@ import { mapMutations, mapState } from "vuex";
export default {
components: {
Modal
Modal,
},
computed: {
...mapState(["session", "modals", "locale"])
...mapState(["session", "modals", "locale"]),
},
methods: {
clearVoteHistory() {
@ -112,11 +104,11 @@ export default {
setRecordVoteHistory() {
this.$store.commit(
"session/setVoteHistoryAllowed",
!this.session.isVoteHistoryAllowed
!this.session.isVoteHistoryAllowed,
);
},
...mapMutations(["toggleModal"])
}
...mapMutations(["toggleModal"]),
},
};
</script>

View file

@ -56,17 +56,17 @@ const faIcons = [
"VolumeMute",
"VoteYea",
"WindowMaximize",
"WindowMinimize"
"WindowMinimize",
];
const fabIcons = ["Github", "Discord"];
library.add(
...faIcons.map(i => fas["fa" + i]),
...fabIcons.map(i => fab["fa" + i])
...faIcons.map((i) => fas["fa" + i]),
...fabIcons.map((i) => fab["fa" + i]),
);
Vue.component("font-awesome-icon", FontAwesomeIcon);
Vue.config.productionTip = false;
new Vue({
render: h => h(App),
store
render: (h) => h(App),
store,
}).$mount("#app");

View file

@ -14,9 +14,9 @@ Vue.use(Vuex);
const getRolesByEdition = (edition = editionJSON.official[0]) => {
return new Map(
rolesJSON
.filter(r => r.edition === edition.id || edition.roles.includes(r.id))
.filter((r) => r.edition === edition.id || edition.roles.includes(r.id))
.sort((a, b) => b.team.localeCompare(a.team))
.map(role => [role.id, role])
.map((role) => [role.id, role]),
);
};
@ -24,35 +24,39 @@ const getTravelersNotInEdition = (edition = editionJSON.official[0]) => {
return new Map(
rolesJSON
.filter(
r =>
(r) =>
r.team === "traveler" &&
r.edition !== edition.id &&
!edition.roles.includes(r.id)
!edition.roles.includes(r.id),
)
.map(role => [role.id, role])
.map((role) => [role.id, role]),
);
};
const set = key => ({ grimoire }, val) => {
grimoire[key] = val;
};
const toggle = key => ({ grimoire }, val) => {
if (val === true || val === false) {
const set =
(key) =>
({ grimoire }, val) => {
grimoire[key] = val;
} else {
grimoire[key] = !grimoire[key];
}
};
};
const clean = id => id.toLocaleLowerCase().replace(/[^a-z0-9]/g, "");
const toggle =
(key) =>
({ grimoire }, val) => {
if (val === true || val === false) {
grimoire[key] = val;
} else {
grimoire[key] = !grimoire[key];
}
};
const clean = (id) => id.toLocaleLowerCase().replace(/[^a-z0-9]/g, "");
// global data maps
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]));
const rolesJSONbyId = new Map(rolesJSON.map((role) => [role.id, role]));
const fabled = new Map(fabledJSON.map((role) => [role.id, role]));
// jinxes
let jinxes = {};
@ -64,8 +68,8 @@ try {
jinxes = new Map(
jinxesJSON.map(({ id, hatred }) => [
clean(id),
new Map(hatred.map(({ id, reason }) => [clean(id), reason]))
])
new Map(hatred.map(({ id, reason }) => [clean(id), reason])),
]),
);
// });
} catch (e) {
@ -87,13 +91,13 @@ const customRole = {
remindersGlobal: [],
setup: false,
team: "townsfolk",
isCustom: true
isCustom: true,
};
export default new Vuex.Store({
modules: {
players,
session
session,
},
state: {
grimoire: {
@ -111,8 +115,8 @@ export default new Vuex.Store({
background: "",
timer: {
name: "",
duration: 0
}
duration: 0,
},
},
modals: {
edition: false,
@ -123,7 +127,7 @@ export default new Vuex.Store({
reminder: false,
role: false,
roles: false,
voteHistory: false
voteHistory: false,
},
edition: editionJSONbyId.get("tb"),
editions: editionJSON,
@ -131,7 +135,7 @@ export default new Vuex.Store({
otherTravelers: getTravelersNotInEdition(),
fabled,
jinxes,
locale
locale,
},
getters: {
/**
@ -146,9 +150,9 @@ export default new Vuex.Store({
const strippedProps = [
"firstNightReminder",
"otherNightReminder",
"isCustom"
"isCustom",
];
roles.forEach(role => {
roles.forEach((role) => {
if (!role.isCustom) {
customRoles.push({ id: role.id });
} else {
@ -167,7 +171,7 @@ export default new Vuex.Store({
});
return customRoles;
},
rolesJSONbyId: () => rolesJSONbyId
rolesJSONbyId: () => rolesJSONbyId,
},
mutations: {
setZoom: set("zoom"),
@ -202,7 +206,7 @@ export default new Vuex.Store({
setCustomRoles(state, roles) {
const processedRoles = roles
// replace numerical role object keys with matching key names
.map(role => {
.map((role) => {
if (role[0]) {
const customKeys = Object.keys(customRole);
const mappedRole = {};
@ -217,19 +221,19 @@ export default new Vuex.Store({
}
})
// clean up role.id
.map(role => {
.map((role) => {
role.id = clean(role.id);
return role;
})
// map existing roles to base definition or pre-populate custom roles to ensure all properties
.map(
role =>
(role) =>
rolesJSONbyId.get(role.id) ||
state.roles.get(role.id) ||
Object.assign({}, customRole, role)
Object.assign({}, customRole, role),
)
// default empty icons and placeholders, clean up firstNight / otherNight
.map(role => {
.map((role) => {
if (rolesJSONbyId.get(role.id)) return role;
role.imageAlt = // map team to generic icon
{
@ -237,32 +241,36 @@ export default new Vuex.Store({
outsider: "outsider",
minion: "minion",
demon: "evil",
fabled: "fabled"
fabled: "fabled",
}[role.team] || "custom";
role.firstNight = Math.abs(role.firstNight);
role.otherNight = Math.abs(role.otherNight);
return role;
})
// 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)
.filter((role) => role.name && role.ability && role.team)
// sort by team
.sort((a, b) => b.team.localeCompare(a.team));
// convert to Map without Fabled
state.roles = new Map(
processedRoles
.filter(role => role.team !== "fabled")
.map(role => [role.id, role])
.filter((role) => role.team !== "fabled")
.map((role) => [role.id, role]),
);
// update Fabled to include custom Fabled from this script
state.fabled = new Map([
...processedRoles.filter(r => r.team === "fabled").map(r => [r.id, r]),
...fabledJSON.map(role => [role.id, role])
...processedRoles
.filter((r) => r.team === "fabled")
.map((r) => [r.id, r]),
...fabledJSON.map((role) => [role.id, role]),
]);
// update extraTravelers map to only show travelers not in this script
state.otherTravelers = new Map(
rolesJSON
.filter(r => r.team === "traveler" && !roles.some(i => i.id === r.id))
.map(role => [role.id, role])
.filter(
(r) => r.team === "traveler" && !roles.some((i) => i.id === r.id),
)
.map((role) => [role.id, role]),
);
},
setEdition(state, edition) {
@ -274,7 +282,7 @@ export default new Vuex.Store({
state.edition = edition;
}
state.modals.edition = false;
}
},
},
plugins: [persistence, socket]
plugins: [persistence, socket],
});

View file

@ -3,7 +3,7 @@
"id": "doomsayer",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": [],
"reminders": ["Used"],
"setup": false,
"name": "Doomsayer",
"team": "fabled",

View file

@ -31,7 +31,7 @@
"hatred": [
{
"id": "Heretic",
"reason": "A Pit-Hag can not create a Heretic. "
"reason": "A Pit-Hag cannot create a Heretic. "
},
{
"id": "Damsel",
@ -39,7 +39,7 @@
},
{
"id": "Politician",
"reason": "A Pit-hag can not create an evil Politician."
"reason": "A Pit-hag cannot create an evil Politician."
}
]
},
@ -69,19 +69,19 @@
},
{
"id": "Ravenkeeper",
"reason": "If Leviathan is in play & the Ravenkeeper dies by execution, they wake that night to use their ability."
"reason": "If Leviathan is in play and the Ravenkeeper dies by execution, they wake that night to use their ability."
},
{
"id": "Sage",
"reason": "If Leviathan is in play & the Sage dies by execution, they wake that night to use their ability."
"reason": "If Leviathan is in play and the Sage dies by execution, they wake that night to use their ability."
},
{
"id": "Farmer",
"reason": "If Leviathan is in play & a Farmer dies by execution, a good player becomes a Farmer that night."
"reason": "If Leviathan is in play and a Farmer dies by execution, a good player becomes a Farmer that night."
},
{
"id": "Mayor",
"reason": "If Leviathan is in play & no execution occurs on day 5, good wins."
"reason": "If Leviathan is in play and no execution occurs on day 5, good wins."
}
]
},
@ -107,11 +107,11 @@
},
{
"id": "Magician",
"reason": "Only 1 jinxed character can be in play. "
"reason": "Only one jinxed character can be in play. "
},
{
"id": "Scarlet Woman",
"reason": "If there are 5 or more players alive and the player holding the Lil' Monsta token dies, the Scarlet Woman is given the Lil' Monsta token tonight."
"reason": "If there are five or more players alive and the player holding the Lil' Monsta token dies, the Scarlet Woman is given the Lil' Monsta token tonight."
}
]
},
@ -120,7 +120,7 @@
"hatred": [
{
"id": "Gambler",
"reason": "If the Lycanthrope is alive and the Gambler kills themself at night, no other players can die tonight."
"reason": "If the Lycanthrope is alive and the Gambler kills themselves at night, no other players can die tonight."
}
]
},
@ -129,11 +129,11 @@
"hatred": [
{
"id": "Engineer",
"reason": "Legion and the Engineer can not both be in play at the start of the game. If the Engineer creates Legion, most players (including all evil players) become evil Legion."
"reason": "Legion and the Engineer cannot both be in play at the start of the game. If the Engineer creates Legion, most players (including all evil players) become evil Legion."
},
{
"id": "Preacher",
"reason": "Only 1 jinxed character can be in play."
"reason": "Only one jinxed character can be in play."
}
]
},
@ -151,11 +151,11 @@
"hatred": [
{
"id": "Magician",
"reason": "When the Spy sees the Grimoire, the Demon and Magician's character tokens are removed."
"reason": "When the Spy sees the Grimoire, the Demon and the Magician's character tokens are removed."
},
{
"id": "Alchemist",
"reason": "The Alchemist can not have the Spy ability."
"reason": "The Alchemist cannot have the Spy ability."
},
{
"id": "Poppy Grower",
@ -163,11 +163,11 @@
},
{
"id": "Damsel",
"reason": "Only 1 jinxed character can be in play. "
"reason": "Only one jinxed character can be in play. "
},
{
"id": "Heretic",
"reason": "Only 1 jinxed character can be in play."
"reason": "Only one jinxed character can be in play."
}
]
},
@ -176,7 +176,7 @@
"hatred": [
{
"id": "Magician",
"reason": "When the Widow sees the Grimoire, the Demon and Magician's character tokens are removed."
"reason": "When the Widow sees the Grimoire, the Demon and the Magician's character tokens are removed."
},
{
"id": "Poppy Grower",
@ -184,15 +184,15 @@
},
{
"id": "Alchemist",
"reason": "The Alchemist can not have the Widow ability."
"reason": "The Alchemist cannot have the Widow ability."
},
{
"id": "Damsel",
"reason": "Only 1 jinxed character can be in play."
"reason": "Only one jinxed character can be in play."
},
{
"id": "Heretic",
"reason": "Only 1 jinxed character can be in play."
"reason": "Only one jinxed character can be in play."
}
]
},
@ -201,7 +201,7 @@
"hatred": [
{
"id": "Heretic",
"reason": "Only 1 jinxed character can be in play."
"reason": "Only one jinxed character can be in play."
}
]
},
@ -210,7 +210,7 @@
"hatred": [
{
"id": "Heretic",
"reason": "The Baron might only add 1 Outsider, not 2."
"reason": "The Baron might only add one Outsider, not two."
}
]
},
@ -219,7 +219,7 @@
"hatred": [
{
"id": "Lil' Monsta",
"reason": "The Marionette neighbors a Minion, not the Demon. The Marionette is not woken to choose who takes the Lil' Monsta token."
"reason": "The Marionette neighbors a Minion, not the Demon. The Marionette is not woken to choose who takes the Lil' Monsta token, and does not learn they are the Marionette if they have the Lil' Monsta token."
},
{
"id": "Poppy Grower",
@ -248,7 +248,7 @@
"hatred": [
{
"id": "Engineer",
"reason": "Riot and the Engineer can not both be in play at the start of the game. \nIf the Engineer creates Riot, the evil players become Riot."
"reason": "Riot and the Engineer cannot both be in play at the start of the game. If the Engineer creates Riot, the evil players become Riot."
},
{
"id": "Golem",
@ -256,7 +256,7 @@
},
{
"id": "Snitch",
"reason": "If the Snitch is in play, each Riot player gets an extra 3 bluffs."
"reason": "If the Snitch is in play, each Riot player gets an extra three bluffs."
},
{
"id": "Saint",
@ -264,19 +264,19 @@
},
{
"id": "Butler",
"reason": "The Butler can not nominate their master."
"reason": "The Butler cannot nominate their master."
},
{
"id": "Pit-Hag",
"reason": "If the Pit-Hag creates Riot, all evil players become Riot. \nIf the Pit-Hag creates Riot after day 3, the game continues for one more day."
"reason": "If the Pit-Hag creates Riot, all evil players become Riot. If the Pit-Hag creates Riot after day 3, the game continues for one more day."
},
{
"id": "Mayor",
"reason": "If the 3rd day begins with just three players alive, the players may choose (as a group) not to nominate at all. If so (and a Mayor is alive) then the Mayor's team wins."
"reason": "If the third day begins with just three players alive, the players may choose (as a group) not to nominate at all. If so (and a Mayor is alive) the Mayor's team wins."
},
{
"id": "Monk",
"reason": "If a Riot player nominates and kills the Monk-protected-player, the Monk-protected-player does not die."
"reason": "If a Riot player nominates a Monk-protected player, the protected-player does not die."
},
{
"id": "Farmer",
@ -284,7 +284,7 @@
},
{
"id": "Innkeeper",
"reason": "If a Riot player nominates an Innkeeper-protected-player, the Innkeeper-protected-player does not die."
"reason": "If a Riot player nominates an Innkeeper-protected player, the protected-player does not die."
},
{
"id": "Sage",
@ -300,7 +300,7 @@
},
{
"id": "Grandmother",
"reason": "If a Riot player nominates and kills the Grandchild, the Grandmother dies too."
"reason": "If a Riot player nominates and kills the grandchild, the Grandmother dies too."
},
{
"id": "King",
@ -308,31 +308,31 @@
},
{
"id": "Exorcist",
"reason": "Only 1 jinxed character can be in play."
"reason": "Only one jinxed character can be in play."
},
{
"id": "Minstrel",
"reason": "Only 1 jinxed character can be in play."
"reason": "Only one jinxed character can be in play."
},
{
"id": "Flowergirl",
"reason": "Only 1 jinxed character can be in play."
"reason": "Only one jinxed character can be in play."
},
{
"id": "Undertaker",
"reason": "Players that die by nomination register as being executed to the Undertaker."
"reason": "Players that die by nomination register as executed to the Undertaker."
},
{
"id": "Cannibal",
"reason": "Players that die by nomination register as being executed to the Cannibal."
"reason": "Players that die by nomination register as executed to the Cannibal."
},
{
"id": "Pacifist",
"reason": "Players that die by nomination register as being executed to the Pacifist."
"reason": "Players that die by nomination register as executed to the Pacifist."
},
{
"id": "Devil's Advocate",
"reason": "Players that die by nomination register as being executed to the Devil's Advocate."
"reason": "Players that die by nomination register as executed to the Devil's Advocate."
},
{
"id": "Investigator",
@ -365,7 +365,7 @@
},
{
"id": "Slayer",
"reason": "If the Slayer slays the Lleech's host, the host dies. "
"reason": "If the Slayer shoots the Lleech's host, the host dies. "
},
{
"id": "Heretic",

View file

@ -1725,7 +1725,7 @@
"Day 4",
"Day 5"],
"setup": false,
"ability": "If more than 1 good player is executed, you win. All players know you are in play. After day 5, evil wins."
"ability": "If more than 1 good player is executed, evil wins. All players know you are in play. After day 5, evil wins."
},
{
"id": "riot",

View file

@ -70,7 +70,9 @@
"customNote": "Add a custom reminder node"
},
"vote":{
"nominated": "nominated",
"nominates": "nominates",
"callexile": "calls for the exile of",
"exclam": "!",
"votes": "votes",
"inFavor": "in favor",
"majorityIs": "majority is",
@ -205,6 +207,7 @@
"reference": {
"title": "Character Reference",
"jinxed": "Jinxed",
"notfirstnight": "*Not the first night",
"teamNames": {
"townsfolk": "townfolk",
"outsider": "outsider",

View file

@ -5,7 +5,7 @@
"team": "fabled",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": [],
"reminders": ["Utilisé"],
"setup": false,
"ability": "Si 4 joueurs ou plus sont en vie, chaque joueur vivant peut, une fois par partie, décider qu'un joueur de son propre alignement meure."
},

View file

@ -70,7 +70,9 @@
"customNote": "Ajouter une note personnalisée"
},
"vote":{
"nominated": "accuse",
"nominates": "accuse",
"callexile": "veut exiler",
"exclam": " !",
"votes": "votes",
"inFavor": "pour",
"majorityIs": "majorité à",
@ -205,6 +207,7 @@
"reference": {
"title": "Réference de rôles",
"jinxed": "Jinx",
"notfirstnight": "* Pas la première nuit",
"teamNames": {
"townsfolk": "villageois",
"outsider": "étranger",

View file

@ -5,22 +5,22 @@ const NEWPLAYER = {
reminders: [],
isVoteless: false,
isDead: false,
pronouns: ""
pronouns: "",
};
const state = () => ({
players: [],
fabled: [],
bluffs: []
bluffs: [],
});
const getters = {
alive({ players }) {
return players.filter(player => !player.isDead).length;
return players.filter((player) => !player.isDead).length;
},
nonTravelers({ players }) {
const nonTravelers = players.filter(
player => player.role.team !== "traveler"
(player) => player.role.team !== "traveler",
);
return Math.min(nonTravelers.length, 15);
},
@ -36,7 +36,7 @@ const getters = {
otherNight.push(role.otherNight);
}
});
fabled.forEach(role => {
fabled.forEach((role) => {
if (role.firstNight && !firstNight.includes(role.firstNight)) {
firstNight.push(role.firstNight);
}
@ -47,32 +47,32 @@ const getters = {
firstNight.sort((a, b) => a - b);
otherNight.sort((a, b) => a - b);
const nightOrder = new Map();
players.forEach(player => {
players.forEach((player) => {
const first = Math.max(firstNight.indexOf(player.role.firstNight), 0);
const other = Math.max(otherNight.indexOf(player.role.otherNight), 0);
nightOrder.set(player, { first, other });
});
fabled.forEach(role => {
fabled.forEach((role) => {
const first = Math.max(firstNight.indexOf(role.firstNight), 0);
const other = Math.max(otherNight.indexOf(role.otherNight), 0);
nightOrder.set(role, { first, other });
});
return nightOrder;
}
},
};
const actions = {
randomize({ state, commit }) {
const players = state.players
.map(a => [Math.random(), a])
.map((a) => [Math.random(), a])
.sort((a, b) => a[0] - b[0])
.map(a => a[1]);
.map((a) => a[1]);
commit("set", players);
},
clearRoles({ state, commit, rootState }) {
let players;
if (rootState.session.isSpectator) {
players = state.players.map(player => {
players = state.players.map((player) => {
if (player.role.team !== "traveler") {
player.role = {};
}
@ -84,13 +84,13 @@ const actions = {
...NEWPLAYER,
name,
id,
pronouns
pronouns,
}));
commit("setFabled", { fabled: [] });
}
commit("set", players);
commit("setBluff");
}
},
};
const mutations = {
@ -119,7 +119,7 @@ const mutations = {
add(state, name) {
state.players.push({
...NEWPLAYER,
name
name,
});
},
remove(state, index) {
@ -128,7 +128,7 @@ const mutations = {
swap(state, [from, to]) {
[state.players[from], state.players[to]] = [
state.players[to],
state.players[from]
state.players[from],
];
// hack: "modify" the array so that Vue notices something changed
state.players.splice(0, 0);
@ -153,7 +153,7 @@ const mutations = {
state.fabled = fabled;
}
}
}
},
};
export default {
@ -161,5 +161,5 @@ export default {
state,
getters,
actions,
mutations
mutations,
};

View file

@ -29,7 +29,7 @@ const state = () => ({
voteHistory: [],
markedPlayer: -1,
isVoteHistoryAllowed: true,
isRolesDistributed: false
isRolesDistributed: false,
});
const getters = {};
@ -37,7 +37,7 @@ const getters = {};
const actions = {};
// mutations helper functions
const set = key => (state, val) => {
const set = (key) => (state, val) => {
state[key] = val;
};
@ -62,7 +62,7 @@ const mutations = {
},
nomination(
state,
{ nomination, votes, votingSpeed, lockedVote, isVoteInProgress } = {}
{ nomination, votes, votingSpeed, lockedVote, isVoteInProgress } = {},
) {
state.nomination = nomination || false;
state.votes = votes || [];
@ -91,14 +91,14 @@ const mutations = {
: gameInfo.state.locale.modal.voteHistory.execution +
(organGrinder && !state.isSpectator ? "*" : ""),
majority: Math.ceil(
players.filter(player => !player.isDead || isExile).length / 2
players.filter((player) => !player.isDead || isExile).length / 2,
),
votes:
organGrinder && state.isSpectator
? null
: players
.filter((player, index) => state.votes[index])
.map(({ name }) => name)
.map(({ name }) => name),
});
},
clearVoteHistory(state) {
@ -114,7 +114,7 @@ const mutations = {
voteSync: handleVote,
lockVote(state, lock) {
state.lockedVote = lock !== undefined ? lock : state.lockedVote + 1;
}
},
};
export default {
@ -122,5 +122,5 @@ export default {
state,
getters,
actions,
mutations
mutations,
};

View file

@ -1,5 +1,5 @@
module.exports = store => {
const updatePagetitle = isPublic =>
module.exports = (store) => {
const updatePagetitle = (isPublic) =>
(document.title = `Blood on the Clocktower ${
isPublic ? "Town Square" : "Grimoire"
}`);
@ -39,27 +39,27 @@ module.exports = store => {
JSON.parse(localStorage.bluffs).forEach((role, index) => {
store.commit("players/setBluff", {
index,
role: store.state.roles.get(role) || {}
role: store.state.roles.get(role) || {},
});
});
}
if (localStorage.fabled !== undefined) {
store.commit("players/setFabled", {
fabled: JSON.parse(localStorage.fabled).map(
fabled => store.state.fabled.get(fabled.id) || fabled
)
(fabled) => store.state.fabled.get(fabled.id) || fabled,
),
});
}
if (localStorage.players) {
store.commit(
"players/set",
JSON.parse(localStorage.players).map(player => ({
JSON.parse(localStorage.players).map((player) => ({
...player,
role:
store.state.roles.get(player.role) ||
store.getters.rolesJSONbyId.get(player.role) ||
{}
}))
{},
})),
);
}
/**** Session related data *****/
@ -141,17 +141,17 @@ module.exports = store => {
case "players/setBluff":
localStorage.setItem(
"bluffs",
JSON.stringify(state.players.bluffs.map(({ id }) => id))
JSON.stringify(state.players.bluffs.map(({ id }) => id)),
);
break;
case "players/setFabled":
localStorage.setItem(
"fabled",
JSON.stringify(
state.players.fabled.map(fabled =>
fabled.isCustom ? fabled : { id: fabled.id }
)
)
state.players.fabled.map((fabled) =>
fabled.isCustom ? fabled : { id: fabled.id },
),
),
);
break;
case "players/add":
@ -165,12 +165,12 @@ module.exports = store => {
localStorage.setItem(
"players",
JSON.stringify(
state.players.players.map(player => ({
state.players.players.map((player) => ({
...player,
// simplify the stored data
role: player.role.id || {}
}))
)
role: player.role.id || {},
})),
),
);
} else {
localStorage.removeItem("players");
@ -180,7 +180,7 @@ module.exports = store => {
if (payload) {
localStorage.setItem(
"session",
JSON.stringify([state.session.isSpectator, payload])
JSON.stringify([state.session.isSpectator, payload]),
);
} else {
localStorage.removeItem("session");

View file

@ -28,11 +28,11 @@ class LiveSession {
this._wss +
channel +
"/" +
(this._isSpectator ? this._store.state.session.playerId : "host")
(this._isSpectator ? this._store.state.session.playerId : "host"),
);
this._socket.addEventListener("message", this._handleMessage.bind(this));
this._socket.onopen = this._onOpen.bind(this);
this._socket.onclose = err => {
this._socket.onclose = (err) => {
this._socket = null;
clearInterval(this._pingTimer);
this._pingTimer = null;
@ -41,7 +41,7 @@ class LiveSession {
this._store.commit("session/setReconnecting", true);
this._reconnectTimer = setTimeout(
() => this.connect(channel),
3 * 1000
3 * 1000,
);
} else {
this._store.commit("session/setSessionId", "");
@ -87,7 +87,7 @@ class LiveSession {
this._sendDirect(
"host",
"getGamestate",
this._store.state.session.playerId
this._store.state.session.playerId,
);
} else {
this.sendGamestate();
@ -105,7 +105,7 @@ class LiveSession {
this._isSpectator
? this._store.state.session.playerId
: Object.keys(this._players).length,
"latency"
"latency",
]);
clearTimeout(this._pingTimer);
this._pingTimer = setTimeout(this._ping.bind(this), this._pingInterval);
@ -151,7 +151,7 @@ class LiveSession {
// create vote history record
this._store.commit(
"session/addHistory",
this._store.state.players.players
this._store.state.players.players,
);
}
this._store.commit("session/nomination", { nomination: params });
@ -229,9 +229,7 @@ class LiveSession {
if (!this._store.state.session.playerId) {
this._store.commit(
"session/setPlayerId",
Math.random()
.toString(36)
.substr(2)
Math.random().toString(36).substr(2),
);
}
this._pings = {};
@ -267,7 +265,7 @@ class LiveSession {
*/
sendGamestate(playerId = "", isLightweight = false) {
if (this._isSpectator) return;
this._gamestate = this._store.state.players.players.map(player => ({
this._gamestate = this._store.state.players.players.map((player) => ({
name: player.name,
id: player.id,
isDead: player.isDead,
@ -275,12 +273,12 @@ class LiveSession {
pronouns: player.pronouns,
...(player.role && player.role.team === "traveler"
? { roleId: player.role.id }
: {})
: {}),
}));
if (isLightweight) {
this._sendDirect(playerId, "gs", {
gamestate: this._gamestate,
isLightweight
isLightweight,
});
} else {
const { session, grimoire } = this._store.state;
@ -298,8 +296,8 @@ class LiveSession {
lockedVote: session.lockedVote,
isVoteInProgress: session.isVoteInProgress,
markedPlayer: session.markedPlayer,
fabled: fabled.map(f => (f.isCustom ? f : { id: f.id })),
...(session.nomination ? { votes: session.votes } : {})
fabled: fabled.map((f) => (f.isCustom ? f : { id: f.id })),
...(session.nomination ? { votes: session.votes } : {}),
});
}
}
@ -325,7 +323,7 @@ class LiveSession {
lockedVote,
isVoteInProgress,
markedPlayer,
fabled
fabled,
} = data;
const players = this._store.state.players.players;
// adjust number of players
@ -343,7 +341,7 @@ class LiveSession {
const player = players[x];
const { roleId } = state;
// update relevant properties
["name", "id", "isDead", "isVoteless", "pronouns"].forEach(property => {
["name", "id", "isDead", "isVoteless", "pronouns"].forEach((property) => {
const value = state[property];
if (player[property] !== value) {
this._store.commit("players/update", { player, property, value });
@ -358,14 +356,14 @@ class LiveSession {
this._store.commit("players/update", {
player,
property: "role",
value: role
value: role,
});
}
} else if (!roleId && player.role.team === "traveler") {
this._store.commit("players/update", {
player,
property: "role",
value: {}
value: {},
});
}
});
@ -380,11 +378,11 @@ class LiveSession {
votes,
votingSpeed,
lockedVote,
isVoteInProgress
isVoteInProgress,
});
this._store.commit("session/setMarkedPlayer", markedPlayer);
this._store.commit("players/setFabled", {
fabled: fabled.map(f => this._store.state.fabled.get(f.id) || f)
fabled: fabled.map((f) => this._store.state.fabled.get(f.id) || f),
});
}
}
@ -402,7 +400,7 @@ class LiveSession {
}
this._sendDirect(playerId, "edition", {
edition: edition.isOfficial ? { id: edition.id } : edition,
...(roles ? { roles } : {})
...(roles ? { roles } : {}),
});
}
@ -427,7 +425,7 @@ class LiveSession {
alert(
`This session contains custom characters that can't be found. ` +
`Please load them before joining! ` +
`Missing roles: ${missing.join(", ")}`
`Missing roles: ${missing.join(", ")}`,
);
this.disconnect();
this._store.commit("toggleModal", "edition");
@ -443,7 +441,7 @@ class LiveSession {
const { fabled } = this._store.state.players;
this._send(
"fabled",
fabled.map(f => (f.isCustom ? f : { id: f.id }))
fabled.map((f) => (f.isCustom ? f : { id: f.id })),
);
}
@ -455,7 +453,7 @@ class LiveSession {
_updateFabled(fabled) {
if (!this._isSpectator) return;
this._store.commit("players/setFabled", {
fabled: fabled.map(f => this._store.state.fabled.get(f.id) || f)
fabled: fabled.map((f) => this._store.state.fabled.get(f.id) || f),
});
}
@ -475,7 +473,7 @@ class LiveSession {
this._send("player", {
index,
property,
value: value.id
value: value.id,
});
} else if (this._gamestate[index].roleId) {
// player was previously a traveler
@ -505,7 +503,7 @@ class LiveSession {
this._store.commit("players/update", {
player,
property: "role",
value: {}
value: {},
});
} else {
// load role, first from session, the global, then fail gracefully
@ -516,7 +514,7 @@ class LiveSession {
this._store.commit("players/update", {
player,
property: "role",
value: role
value: role,
});
}
} else {
@ -556,7 +554,7 @@ class LiveSession {
player,
property: "pronouns",
value,
isFromSockets: true
isFromSockets: true,
});
}
@ -577,12 +575,12 @@ class LiveSession {
}
}
// remove claimed seats from players that are no longer connected
this._store.state.players.players.forEach(player => {
this._store.state.players.players.forEach((player) => {
if (player.id && !this._players[player.id]) {
this._store.commit("players/update", {
player,
property: "id",
value: ""
value: "",
});
}
});
@ -596,7 +594,7 @@ class LiveSession {
const pings = Object.values(this._pings);
this._store.commit(
"session/setPing",
Math.round(pings.reduce((a, b) => a + b, 0) / pings.length)
Math.round(pings.reduce((a, b) => a + b, 0) / pings.length),
);
}
}
@ -608,7 +606,7 @@ class LiveSession {
if (!this._isSpectator || playerIdOrCount) {
this._store.commit(
"session/setPlayerCount",
this._isSpectator ? playerIdOrCount : Object.keys(this._players).length
this._isSpectator ? playerIdOrCount : Object.keys(this._players).length,
);
}
}
@ -623,7 +621,7 @@ class LiveSession {
delete this._players[playerId];
this._store.commit(
"session/setPlayerCount",
Object.keys(this._players).length
Object.keys(this._players).length,
);
}
@ -656,7 +654,7 @@ class LiveSession {
this._store.commit("players/update", {
player: players[oldIndex],
property,
value: ""
value: "",
});
}
// add playerId to new seat
@ -680,7 +678,7 @@ class LiveSession {
if (player.id && player.role) {
message[player.id] = [
"player",
{ index, property: "role", value: player.role.id }
{ index, property: "role", value: player.role.id },
];
}
});
@ -755,7 +753,7 @@ class LiveSession {
if (this._isSpectator) return;
this._send(
"isVoteHistoryAllowed",
this._store.state.session.isVoteHistoryAllowed
this._store.state.session.isVoteHistoryAllowed,
);
}
@ -802,7 +800,7 @@ class LiveSession {
this._send("vote", [
index,
this._store.state.session.votes[index],
!this._isSpectator
!this._isSpectator,
]);
}
}
@ -881,7 +879,7 @@ class LiveSession {
}
}
export default store => {
export default (store) => {
// setup
const session = new LiveSession(store);

View file

@ -1,5 +1,5 @@
module.exports = {
// if the app is supposed to run on Github Pages in a subfolder, use the following config:
// publicPath: process.env.NODE_ENV === "production" ? "/townsquare/" : "/"
publicPath: process.env.NODE_ENV === "production" ? "/" : "/"
publicPath: process.env.NODE_ENV === "production" ? "/" : "/",
};