Merge branch 'develop'

This commit is contained in:
Pingumask 2024-04-19 23:26:51 +00:00
commit 498990d21e
36 changed files with 1747 additions and 627 deletions

View file

@ -8,9 +8,9 @@ module.exports = {
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",
"no-console": process.env.NODE_ENV === "production" ? 1 : 0,
"no-debugger": process.env.NODE_ENV === "production" ? 1 : 0,
"vue/multi-word-component-names": 0,
"vue/no-reserved-component-names": 0,
},
};

View file

@ -1,6 +1,11 @@
# Release Notes
## Upcomming Version
- Adding a missing jinx
- Updating night order (and its print)
- Correcting automatic adding/deletion of Fabled
- Adding all missing roles (up to Summoner)
- Correcting Dawn night order
### Version 3.17.0

134
package-lock.json generated
View file

@ -31,7 +31,7 @@
"prettier": "^3.0.3"
},
"engines": {
"node": "^16"
"node": "^18"
}
},
"node_modules/@aashutoshrathi/word-wrap": {
@ -57,11 +57,11 @@
}
},
"node_modules/@babel/code-frame": {
"version": "7.22.13",
"resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.22.13.tgz",
"integrity": "sha512-XktuhWlJ5g+3TJXc5upd9Ks1HutSArik6jf2eAjYFyIOf4ej3RN+184cZbzDvbPnuTJIUhPKKJE3cIsYTiAT3w==",
"version": "7.23.4",
"resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.23.4.tgz",
"integrity": "sha512-r1IONyb6Ia+jYR2vvIDhdWdlTGhqbBoFqLTQidzZ4kepUFH15ejXvFHxCVbtl7BOXIudsIubf4E81xeA3h3IXA==",
"dependencies": {
"@babel/highlight": "^7.22.13",
"@babel/highlight": "^7.23.4",
"chalk": "^2.4.2"
},
"engines": {
@ -172,9 +172,9 @@
}
},
"node_modules/@babel/highlight": {
"version": "7.22.20",
"resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.22.20.tgz",
"integrity": "sha512-dkdMCN3py0+ksCgYmGG8jKeGA/8Tk+gJwSYYlFGxG5lmhfKNoAy004YpLxpS1W2J8m/EK2Ew+yOs9pVRwO89mg==",
"version": "7.23.4",
"resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.23.4.tgz",
"integrity": "sha512-acGdbYSfp2WheJoJm/EBBBLh/ID8KDc64ISZ9DYtBmC8/Q204PZJLHyzeB5qMzJ5trcOkybd78M4x2KWsUq++A==",
"dependencies": {
"@babel/helper-validator-identifier": "^7.22.20",
"chalk": "^2.4.2",
@ -249,9 +249,9 @@
}
},
"node_modules/@babel/parser": {
"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==",
"version": "7.23.4",
"resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.23.4.tgz",
"integrity": "sha512-vf3Xna6UEprW+7t6EtOmFpHNAuxw3xqPZghy+brsnusscJRW5BMUzzHZc5ICjULee81WeUV2jjakG09MDglJXQ==",
"bin": {
"parser": "bin/babel-parser.js"
},
@ -315,9 +315,9 @@
}
},
"node_modules/@eslint/js": {
"version": "8.53.0",
"resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.53.0.tgz",
"integrity": "sha512-Kn7K8dx/5U6+cT1yEhpX1w4PCSg0M+XyRILPgvwcEBjerFWCwQj5sbr3/VmxqV0JGHCBCzyd6LxypEuehypY1w==",
"version": "8.54.0",
"resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.54.0.tgz",
"integrity": "sha512-ut5V+D+fOoWPgGGNj83GGjnntO39xDy6DWxO0wb7Jp3DcMX0TfIqdzHF85VTQkerdyGmuuMD9AKAo5KiNlf/AQ==",
"dev": true,
"engines": {
"node": "^12.22.0 || ^14.17.0 || >=16.0.0"
@ -739,9 +739,9 @@
"integrity": "sha512-hov8bUuiLiyFPGyFPE1lwWhmzYbirOXQNNo40+y3zow8aFVTeyn3VWL0VFFfdNddA8S4Vf0Tc062rzyNr7Paag=="
},
"node_modules/@types/node": {
"version": "20.9.0",
"resolved": "https://registry.npmjs.org/@types/node/-/node-20.9.0.tgz",
"integrity": "sha512-nekiGu2NDb1BcVofVcEKMIwzlx4NjHlcjhoxxKBNLtz15Y1z7MYf549DFvkHSId02Ax6kGwWntIBPC3l/JZcmw==",
"version": "20.9.2",
"resolved": "https://registry.npmjs.org/@types/node/-/node-20.9.2.tgz",
"integrity": "sha512-WHZXKFCEyIUJzAwh3NyyTHYSR35SevJ6mZ1nWwJafKtiQbqRTIKSRcw3Ma3acqgsent3RRDqeVwpHntMk+9irg==",
"dependencies": {
"undici-types": "~5.26.4"
}
@ -1860,9 +1860,9 @@
}
},
"node_modules/caniuse-lite": {
"version": "1.0.30001561",
"resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001561.tgz",
"integrity": "sha512-NTt0DNoKe958Q0BE0j0c1V9jbUzhBxHIEJy7asmGrpE0yG63KTV7PLHPnK2E1O9RsQrQ081I3NLuXGS6zht3cw==",
"version": "1.0.30001563",
"resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001563.tgz",
"integrity": "sha512-na2WUmOxnwIZtwnFI2CZ/3er0wdNzU7hN+cPYz/z2ajHThnkWjNBOpEPP4n+4r2WPM847JaMotaJE3bnfzjyKw==",
"funding": [
{
"type": "opencollective",
@ -2549,6 +2549,11 @@
"resolved": "https://registry.npmjs.org/de-indent/-/de-indent-1.0.2.tgz",
"integrity": "sha512-e/1zu3xH5MQryN2zdVaF0OrdNLUbvWxzMbi+iNA6Bky7l1RoP8a2fIbRocyHclXt/arDrrR6lL3TqFD9pMQTsg=="
},
"node_modules/debounce": {
"version": "1.2.1",
"resolved": "https://registry.npmjs.org/debounce/-/debounce-1.2.1.tgz",
"integrity": "sha512-XRRe6Glud4rd/ZGQfiV1ruXSfbvfJedlV9Y6zOlP+2K04vBYiJEte6stfFkCP03aMnY5tsipamumUjL14fofug=="
},
"node_modules/debug": {
"version": "4.3.4",
"resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
@ -2989,9 +2994,9 @@
"integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="
},
"node_modules/electron-to-chromium": {
"version": "1.4.581",
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.581.tgz",
"integrity": "sha512-6uhqWBIapTJUxgPTCHH9sqdbxIMPt7oXl0VcAL1kOtlU6aECdcMncCrX5Z7sHQ/invtrC9jUQUef7+HhO8vVFw=="
"version": "1.4.589",
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.589.tgz",
"integrity": "sha512-zF6y5v/YfoFIgwf2dDfAqVlPPsyQeWNpEWXbAlDUS8Ax4Z2VoiiZpAPC0Jm9hXEkJm2vIZpwB6rc4KnLTQffbQ=="
},
"node_modules/emoji-regex": {
"version": "8.0.0",
@ -3059,9 +3064,9 @@
}
},
"node_modules/es-module-lexer": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.4.0.tgz",
"integrity": "sha512-lcCr3v3OLezdfFyx9r5NRYHOUTQNnFEQ9E87Mx8Kc+iqyJNkO7MJoB4GQRTlIMw9kLLTwGw0OAkm4BQQud/d9g=="
"version": "1.4.1",
"resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.4.1.tgz",
"integrity": "sha512-cXLGjP0c4T3flZJKQSuziYoq7MlT+rnvfZjfp7h+I7K9BNX54kP9nyWvdbwjQ4u1iWbOL4u96fgeZLToQlZC7w=="
},
"node_modules/escalade": {
"version": "3.1.1",
@ -3088,15 +3093,15 @@
}
},
"node_modules/eslint": {
"version": "8.53.0",
"resolved": "https://registry.npmjs.org/eslint/-/eslint-8.53.0.tgz",
"integrity": "sha512-N4VuiPjXDUa4xVeV/GC/RV3hQW9Nw+Y463lkWaKKXKYMvmRiRDAtfpuPFLN+E1/6ZhyR8J2ig+eVREnYgUsiag==",
"version": "8.54.0",
"resolved": "https://registry.npmjs.org/eslint/-/eslint-8.54.0.tgz",
"integrity": "sha512-NY0DfAkM8BIZDVl6PgSa1ttZbx3xHgJzSNJKYcQglem6CppHyMhRIQkBVSSMaSRnLhig3jsDbEzOjwCVt4AmmA==",
"dev": true,
"dependencies": {
"@eslint-community/eslint-utils": "^4.2.0",
"@eslint-community/regexpp": "^4.6.1",
"@eslint/eslintrc": "^2.1.3",
"@eslint/js": "8.53.0",
"@eslint/js": "8.54.0",
"@humanwhocodes/config-array": "^0.11.13",
"@humanwhocodes/module-importer": "^1.0.1",
"@nodelib/fs.walk": "^1.2.8",
@ -3774,9 +3779,9 @@
}
},
"node_modules/flat-cache": {
"version": "3.1.1",
"resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.1.1.tgz",
"integrity": "sha512-/qM2b3LUIaIgviBQovTLvijfyOQXPtSRnRK26ksj2J7rzPIecePUIpJsZ4T02Qg+xiAEKIs5K8dsHEd+VaKa/Q==",
"version": "3.2.0",
"resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz",
"integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==",
"dev": true,
"dependencies": {
"flatted": "^3.2.9",
@ -3784,7 +3789,7 @@
"rimraf": "^3.0.2"
},
"engines": {
"node": ">=12.0.0"
"node": "^10.12.0 || >=12.0.0"
}
},
"node_modules/flatted": {
@ -4159,6 +4164,11 @@
}
]
},
"node_modules/html-escaper": {
"version": "2.0.2",
"resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz",
"integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg=="
},
"node_modules/html-minifier-terser": {
"version": "6.1.0",
"resolved": "https://registry.npmjs.org/html-minifier-terser/-/html-minifier-terser-6.1.0.tgz",
@ -4331,9 +4341,9 @@
]
},
"node_modules/ignore": {
"version": "5.2.4",
"resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz",
"integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==",
"version": "5.3.0",
"resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.0.tgz",
"integrity": "sha512-g7dmpshy+gD7mh88OC9NwSGTKoc3kyLAZQRU1mt53Aw/vnvfXnbC+F/7F7QoYVKbV+KNvJx8wArewKy1vXMtlg==",
"engines": {
"node": ">= 4"
}
@ -4832,31 +4842,11 @@
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
},
"node_modules/lodash.debounce": {
"version": "4.0.8",
"resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz",
"integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow=="
},
"node_modules/lodash.defaultsdeep": {
"version": "4.6.1",
"resolved": "https://registry.npmjs.org/lodash.defaultsdeep/-/lodash.defaultsdeep-4.6.1.tgz",
"integrity": "sha512-3j8wdDzYuWO3lM3Reg03MuQR957t287Rpcxp1njpEa8oDrikb+FwGdW3n+FELh/A6qib6yPit0j/pv9G/yeAqA=="
},
"node_modules/lodash.escape": {
"version": "4.0.1",
"resolved": "https://registry.npmjs.org/lodash.escape/-/lodash.escape-4.0.1.tgz",
"integrity": "sha512-nXEOnb/jK9g0DYMr1/Xvq6l5xMD7GDG55+GSYIYmS0G4tBk/hURD4JR9WCavs04t33WmJx9kCyp9vJ+mr4BOUw=="
},
"node_modules/lodash.flatten": {
"version": "4.4.0",
"resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz",
"integrity": "sha512-C5N2Z3DgnnKr0LOpv/hKCgKdb7ZZwafIrsesve6lmzvZIRZRGaZ/l6Q8+2W7NaT+ZwO3fFlSCzCzrDCFdJfZ4g=="
},
"node_modules/lodash.invokemap": {
"version": "4.6.0",
"resolved": "https://registry.npmjs.org/lodash.invokemap/-/lodash.invokemap-4.6.0.tgz",
"integrity": "sha512-CfkycNtMqgUlfjfdh2BhKO/ZXrP8ePOX5lEU/g0R3ItJcnuxWDwokMGKx1hWcfOikmyOVx6X9IwWnDGlgKl61w=="
},
"node_modules/lodash.mapvalues": {
"version": "4.6.0",
"resolved": "https://registry.npmjs.org/lodash.mapvalues/-/lodash.mapvalues-4.6.0.tgz",
@ -4873,21 +4863,11 @@
"integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
"dev": true
},
"node_modules/lodash.pullall": {
"version": "4.2.0",
"resolved": "https://registry.npmjs.org/lodash.pullall/-/lodash.pullall-4.2.0.tgz",
"integrity": "sha512-VhqxBKH0ZxPpLhiu68YD1KnHmbhQJQctcipvmFnqIBDYzcIHzf3Zpu0tpeOKtR4x76p9yohc506eGdOjTmyIBg=="
},
"node_modules/lodash.uniq": {
"version": "4.5.0",
"resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz",
"integrity": "sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ=="
},
"node_modules/lodash.uniqby": {
"version": "4.7.0",
"resolved": "https://registry.npmjs.org/lodash.uniqby/-/lodash.uniqby-4.7.0.tgz",
"integrity": "sha512-e/zcLx6CSbmaEgFHCA7BnoQKyCtKMxnuWrJygbwPs/AIn+IMKl66L8/s+wBUn5LRw2pZx3bUHibiV1b6aTWIww=="
},
"node_modules/log-symbols": {
"version": "4.1.0",
"resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz",
@ -6351,9 +6331,9 @@
}
},
"node_modules/prettier": {
"version": "3.0.3",
"resolved": "https://registry.npmjs.org/prettier/-/prettier-3.0.3.tgz",
"integrity": "sha512-L/4pUDMxcNa8R/EthV08Zt42WBO4h1rarVtK0K+QJG0X187OLo7l699jWw0GKuwzkPQ//jMFA/8Xm6Fh3J/DAg==",
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/prettier/-/prettier-3.1.0.tgz",
"integrity": "sha512-TQLvXjq5IAibjh8EpBIkNKxO749UEWABoiIZehEPiY4GNpVdhaFKqSTu+QrlU6D2dPAfubRmtJTi4K4YkQ5eXw==",
"dev": true,
"bin": {
"prettier": "bin/prettier.cjs"
@ -8139,23 +8119,19 @@
}
},
"node_modules/webpack-bundle-analyzer": {
"version": "4.9.1",
"resolved": "https://registry.npmjs.org/webpack-bundle-analyzer/-/webpack-bundle-analyzer-4.9.1.tgz",
"integrity": "sha512-jnd6EoYrf9yMxCyYDPj8eutJvtjQNp8PHmni/e/ulydHBWhT5J3menXt3HEkScsu9YqMAcG4CfFjs3rj5pVU1w==",
"version": "4.10.1",
"resolved": "https://registry.npmjs.org/webpack-bundle-analyzer/-/webpack-bundle-analyzer-4.10.1.tgz",
"integrity": "sha512-s3P7pgexgT/HTUSYgxJyn28A+99mmLq4HsJepMPzu0R8ImJc52QNqaFYW1Z2z2uIb1/J3eYgaAWVpaC+v/1aAQ==",
"dependencies": {
"@discoveryjs/json-ext": "0.5.7",
"acorn": "^8.0.4",
"acorn-walk": "^8.0.0",
"commander": "^7.2.0",
"debounce": "^1.2.1",
"escape-string-regexp": "^4.0.0",
"gzip-size": "^6.0.0",
"html-escaper": "^2.0.2",
"is-plain-object": "^5.0.0",
"lodash.debounce": "^4.0.8",
"lodash.escape": "^4.0.1",
"lodash.flatten": "^4.4.0",
"lodash.invokemap": "^4.6.0",
"lodash.pullall": "^4.2.0",
"lodash.uniqby": "^4.7.0",
"opener": "^1.5.2",
"picocolors": "^1.0.0",
"sirv": "^2.0.3",

View file

@ -44,6 +44,6 @@
"url": "https://github.com//bra1n/townsquare.git"
},
"engines": {
"node": "^16"
"node": "^18"
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 133 KiB

BIN
src/assets/icons/dawn.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 85 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 112 KiB

BIN
src/assets/icons/dusk.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 79 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 130 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 104 KiB

BIN
src/assets/icons/harpy.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 122 KiB

BIN
src/assets/icons/hatter.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 141 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 85 KiB

BIN
src/assets/icons/kazali.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 89 KiB

BIN
src/assets/icons/knight.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 160 KiB

BIN
src/assets/icons/ojo.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 128 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 114 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 115 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 82 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 42 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 67 KiB

BIN
src/assets/icons/vizier.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 64 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 82 KiB

View file

@ -44,7 +44,7 @@
)})`,
}"
:key="edition.id"
@click="setEdition(edition)"
@click="runEdition(edition)"
>
{{ edition.name }}
</li>
@ -191,16 +191,19 @@ export default {
"setEdition",
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))) {
const fabled = [];
roles.forEach((role) => {
if (this.$store.state.fabled.has(role.id || role)) {
fabled.push(this.$store.state.fabled.get(role.id || role));
}
});
this.$store.commit("players/setFabled", { fabled });
}
// set fabled
const fabled = [];
roles.forEach((role) => {
if (this.$store.state.fabled.has(role.id || role)) {
fabled.push(this.$store.state.fabled.get(role.id || role));
}
});
this.$store.commit("players/setFabled", { fabled });
},
runEdition(edition) {
this.$store.commit("setEdition", edition);
// The editions contain no Fabled
this.$store.commit("players/setFabled", { fabled: [] });
},
...mapMutations(["toggleModal", "setEdition"]),
},

View file

@ -36,6 +36,18 @@
}}</small
>
</span>
<span
class="player"
v-if="
(role.id == 'dawn' || role.team == 'fabled') &&
!session.isSpectator &&
players.length &&
players[0].role.id
"
>
<br />
<small> </small>
</span>
</span>
<span
class="icon"
@ -92,6 +104,20 @@
}}</small
>
</span>
<span
class="player"
v-if="
(role.id == 'dawn' ||
role.id == 'dusk' ||
role.team == 'fabled') &&
!session.isSpectator &&
players.length &&
players[0].role.id
"
>
<br />
<small> </small>
</span>
</span>
<span class="reminder" v-if="role.otherNightReminder">
{{ role.otherNightReminder }}
@ -113,13 +139,54 @@ export default {
computed: {
rolesFirstNight: function () {
const rolesFirstNight = [];
// Ajouter le matin à l'ordre nocturne
rolesFirstNight.push({
id: "dawn",
name: this.locale.modal.nightOrder.dawn,
firstNight: 1000,
team: "default",
players: [],
firstNightReminder: this.locale.modal.nightOrder.dawnDescription1,
});
var toymaker = false;
// Ajout des fabuleux
this.fabled.forEach((fabled) => {
if (fabled.firstNight) {
rolesFirstNight.push(Object.assign({ players: [] }, fabled));
} else if (fabled.id == "toymaker") {
toymaker = true;
}
});
this.roles.forEach((role) => {
const players = this.players.filter((p) => p.role.id === role.id);
if (role.firstNight && role.team !== "traveler") {
rolesFirstNight.push(Object.assign({ players }, role));
}
});
// Ajout des Voyageurs, en n'ajoutant qu'une fois ceux en double
const seenTravelers = [];
var nbTravelers = 0;
this.players.forEach((player) => {
if (player.role.team == "traveler") {
nbTravelers++;
if (!seenTravelers.includes(player.role.id)) {
seenTravelers.push(player.role.id);
if (player.role.firstNight) {
const players = this.players.filter(
(p) => p.role.id === player.role.id,
);
rolesFirstNight.push(Object.assign({ players }, player.role));
}
}
}
});
// Ajouter minion / demon infos à l'ordre nocturne
if (this.players.length > 6) {
if (this.players.length - nbTravelers > 6 || toymaker) {
rolesFirstNight.push(
{
id: "evil",
id: "minion",
name: this.locale.modal.nightOrder.minionInfo,
firstNight: 5,
firstNight: 7,
team: "minion",
players: this.players.filter((p) => p.role.team === "minion"),
firstNightReminder:
@ -128,7 +195,7 @@ export default {
{
id: "evil",
name: this.locale.modal.nightOrder.demonInfo,
firstNight: 8,
firstNight: 10,
team: "demon",
players: this.players.filter((p) => p.role.team === "demon"),
firstNightReminder:
@ -136,37 +203,66 @@ export default {
},
);
}
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) => {
rolesFirstNight.push(Object.assign({ players: [] }, fabled));
});
rolesFirstNight.sort((a, b) => a.firstNight - b.firstNight);
return rolesFirstNight;
},
rolesOtherNight: function () {
const rolesOtherNight = [];
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));
}
});
rolesOtherNight.push(
{
id: "dusk",
name: this.locale.modal.nightOrder.dusk,
team: "default",
otherNight: 1,
players: [],
otherNightReminder: this.locale.modal.nightOrder.duskDescription,
},
{
id: "dawn",
name: this.locale.modal.nightOrder.dawn,
team: "default",
otherNight: 1000,
players: [],
otherNightReminder: this.locale.modal.nightOrder.dawnDescription2,
},
);
this.fabled
.filter(({ otherNight }) => otherNight)
.forEach((fabled) => {
rolesOtherNight.push(Object.assign({ players: [] }, fabled));
});
this.roles.forEach((role) => {
const players = this.players.filter((p) => p.role.id === role.id);
if (role.otherNight && role.team !== "traveler") {
rolesOtherNight.push(Object.assign({ players }, role));
}
});
// Ajout des Voyageurs, en n'ajoutant qu'une fois ceux en double
const seenTravelers = [];
this.players.forEach((player) => {
if (
player.role.otherNight &&
player.role.team == "traveler" &&
!seenTravelers.includes(player.role.id)
) {
const players = this.players.filter(
(p) => p.role.id === player.role.id,
);
seenTravelers.push(player.role.id);
rolesOtherNight.push(Object.assign({ players }, player.role));
}
});
rolesOtherNight.sort((a, b) => a.otherNight - b.otherNight);
return rolesOtherNight;
},
...mapState(["roles", "modals", "edition", "grimoire", "locale"]),
...mapState([
"roles",
"modals",
"edition",
"grimoire",
"locale",
"session",
]),
...mapState("players", ["players", "fabled"]),
},
methods: {
@ -255,15 +351,32 @@ h4 {
}
}
}
.traveler {
.name {
background: linear-gradient(90deg, $traveler, transparent 35%);
.night .other & {
background: linear-gradient(-90deg, $traveler, transparent 35%);
}
}
}
.default {
.name {
background: linear-gradient(90deg, $default, transparent 35%);
.night .other & {
background: linear-gradient(-90deg, $default, transparent 35%);
}
}
}
ul {
li {
display: flex;
width: 100%;
margin-bottom: 3px;
.icon {
width: 6vh;
background-size: cover;
background-position: 0 0;
width: 5vh;
background-size: 100% auto;
background-position: center center;
background-repeat: no-repeat;
flex-grow: 0;
flex-shrink: 0;
text-align: center;
@ -277,7 +390,7 @@ ul {
.name {
flex-grow: 0;
flex-shrink: 0;
width: 15%;
width: 5%;
text-align: right;
font-size: 110%;
padding: 5px;

View file

@ -68,8 +68,8 @@
vote.votes == null
? 'minus-square'
: vote.votes.length >= vote.majority
? 'check-square'
: 'square',
? 'check-square'
: 'square',
]"
/>
</td>

View file

@ -3,7 +3,9 @@
"id": "doomsayer",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Used"],
"reminders": [
"Used"
],
"setup": false,
"name": "Doomsayer",
"team": "fabled",
@ -13,7 +15,10 @@
"id": "angel",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Protect", "Something Bad"],
"reminders": [
"Protect",
"Something Bad"
],
"setup": false,
"name": "Angel",
"team": "fabled",
@ -23,7 +28,9 @@
"id": "buddhist",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Veteran"],
"reminders": [
"Veteran"
],
"setup": false,
"name": "Buddhist",
"team": "fabled",
@ -33,7 +40,9 @@
"id": "hellslibrarian",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Something Bad"],
"reminders": [
"Something Bad"
],
"setup": false,
"name": "Hell's Librarian",
"team": "fabled",
@ -43,7 +52,10 @@
"id": "revolutionary",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Same alignment", "Used"],
"reminders": [
"Same alignment",
"Used"
],
"setup": false,
"name": "Revolutionary",
"team": "fabled",
@ -62,9 +74,11 @@
{
"id": "toymaker",
"firstNightReminder": "",
"otherNight": 1,
"otherNight": 2,
"otherNightReminder": "If it is a night when a Demon attack could end the game, and the Demon is marked “Final night: No Attack,” then the Demon does not act tonight. (Do not wake them.)",
"reminders": ["Final Night: No Attack"],
"reminders": [
"Final Night: No Attack"
],
"setup": false,
"name": "Toymaker",
"team": "fabled",
@ -74,7 +88,9 @@
"id": "fibbin",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Used"],
"reminders": [
"Used"
],
"setup": false,
"name": "Fibbin",
"team": "fabled",
@ -83,9 +99,12 @@
{
"id": "duchess",
"firstNightReminder": "",
"otherNight": 1,
"otherNight": 2,
"otherNightReminder": "Wake each player marked “Visitor” or “False Info” one at a time. Show them the Duchess token, then fingers (1, 2, 3) equaling the number of evil players marked “Visitor” or, if you are waking the player marked “False Info,” show them any number of fingers except the number of evil players marked “Visitor.”",
"reminders": ["Visitor", "False Info"],
"reminders": [
"Visitor",
"False Info"
],
"setup": false,
"name": "Duchess",
"team": "fabled",
@ -105,7 +124,9 @@
"id": "spiritofivory",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["No extra evil"],
"reminders": [
"No extra evil"
],
"setup": false,
"name": "Spirit of Ivory",
"team": "fabled",
@ -127,10 +148,46 @@
"firstNight": 1,
"firstNightReminder": "Mark a good player as \"Safe\". Wake each evil player and show them the marked player.",
"otherNightReminder": "",
"reminders": ["Safe"],
"reminders": [
"Safe"
],
"setup": false,
"name": "Storm Catcher",
"team": "fabled",
"ability": "Name a good character. If in play, they can only die by execution, but evil players learn which player it is."
},
{
"id": "ferryman",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": [],
"setup": false,
"name": "Ferryman",
"team": "fabled",
"ability": "On the final day, all dead players regain their vote token."
},
{
"id": "gardener",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": [
"Assigned"
],
"setup": false,
"name": "Gardener",
"team": "fabled",
"ability": "The Storyteller assigns 1 or more players' characters."
},
{
"id": "bootlegger",
"firstNight": 1,
"firstNightReminder": "If applicable, use homebrew rule(s).",
"otherNight": 2,
"otherNightReminder": "If applicable, use homebrew rule(s).",
"reminders": [],
"setup": false,
"name": "Bootlegger",
"team": "fabled",
"ability": "This script has homebrew characters or rules."
}
]

View file

@ -40,6 +40,10 @@
{
"id": "Politician",
"reason": "A Pit-hag cannot create an evil Politician."
},
{
"id": "Village Idiot",
"reason": "If there is a spare token, the Pit-Hag can create an extra Village Idiot. If so, the drunk Village Idiot might change."
}
]
},
@ -82,6 +86,10 @@
{
"id": "Mayor",
"reason": "If Leviathan is in play and no execution occurs on day 5, good wins."
},
{
"id": "Pit-Hag",
"reason": "After day 5, the Pit-Hag cannot choose Leviathan."
}
]
},
@ -382,7 +390,7 @@
},
{
"id": "Flowergirl",
"reason": "If players' eyes were closed during the nominations, the Flowergirl learns how many times the Demon voted."
"reason": "If players' eyes were closed during the nominations, the Flowergirl learns how many times the Demon voted."
},
{
"id": "Lil' Monsta",
@ -397,5 +405,163 @@
"reason": "Only 1 jinxed character can be in play. Evil players start knowing which player and character it is."
}
]
},
{
"id": "Yaggababble",
"hatred": [
{
"id": "Exorcist",
"reason": "If the Exorcist chooses the Yaggababble, the Yaggababble ability does not kill tonight."
}
]
},
{
"id": "Plague Doctor",
"hatred": [
{
"id": "Baron",
"reason": "If the Storyteller gains the Baron ability, up to two players become out-of-play Outsiders."
},
{
"id": "Boomdandy",
"reason": "If the Plague Doctor is executed and the Storyteller would gain the Boomdandy ability, the Boomdandy ability triggers immediately."
},
{
"id": "Evil Twin",
"reason": "The Storyteller cannot gain the Evil Twin ability if the Plague Doctor dies."
},
{
"id": "Fearmonger",
"reason": "If the Plague Doctor dies, a living Minion gains the Fearmonger ability in addition to their own ability, and learns this."
},
{
"id": "Goblin",
"reason": "If the Plague Doctor dies, a living Minion gains the Goblin ability in addition to their own ability, and learns this."
},
{
"id": "Scarlet Woman",
"reason": "If the Plague Doctor dies, a living Minion gains the Scarlet Woman ability in addition to their own ability, and learns this."
},
{
"id": "Spy",
"reason": "If the Plague Doctor dies, a living Minion gains the Spy ability in addition to their own ability, and learns this."
},
{
"id": "Marionette",
"reason": "If the Demon has a neighbor who is alive and a Townsfolk or Outsider when the Plague Doctor dies, that player becomes an evil Marionette. If there is already an extra evil player, this does not happen."
}
]
},
{
"id": "Summoner",
"hatred": [
{
"id": "Alchemist",
"reason": "If there is an Alchemist-Summoner in play, the game starts with a Demon in play, as normal. If the Alchemist-Summoner chooses a player, they make that player a Demon but do not change their alignment."
},
{
"id": "Clockmaker",
"reason": "If the Summoner is in play, the Clockmaker does not receive their information until a Demon is created."
},
{
"id": "Poppy Grower",
"reason": "If the Poppy Grower is alive when the Summoner acts, the Summoner chooses which Demon, but the Storyteller chooses which player."
},
{
"id": "Marionette",
"reason": "The Marionette neighbours the Summoner. The Summoner knows who the Marionette is."
},
{
"id": "Kazali",
"reason": "The Kazali can not choose to create a Summoner."
},
{
"id": "Legion",
"reason": "If the Summoner creates Legion, most players (including all evil players) become evil Legion."
},
{
"id": "Riot",
"reason": "If the Summoner creates Riot, the chosen player and all evil players become Riot. The chosen player must be one of the Summoner's good living neighbours."
}
]
},
{
"id": "Vizier",
"hatred": [
{
"id": "Magician",
"reason": "Only 1 jinxed character can be in play. Evil players start knowing which player and character it is."
},
{
"id": "Alchemist",
"reason": "If the Alchemist has the Vizier ability, they may only choose to execute immediately if three or more players voted, regardless of those players' alignment."
},
{
"id": "Courtier",
"reason": "If the Vizier loses their ability, they learn this and if the Vizier is executed while they have their ability, their team wins."
},
{
"id": "Preacher",
"reason": "If the Vizier loses their ability, they learn this and if the Vizier is executed while they have their ability, their team wins."
},
{
"id": "Investigator",
"reason": "If the Investigator learns that the Vizier is in play, the existence of the Vizier is not announced by the Storyteller."
},
{
"id": "Fearmonger",
"reason": "The Vizier wakes with the Fearmonger, learns who they choose and cannot choose to execute that player."
},
{
"id": "Lil' Monsta",
"reason": "The Vizier can die by execution if they are babysitting Lil' Monsta."
}
]
},
{
"id": "Hatter",
"hatred": [
{
"id": "Legion",
"reason": "If the Hatter dies and Legion is in play, nothing happens. If the Hatter dies and an evil player chooses Legion, all current evil players become Legion."
},
{
"id": "Leviathan",
"reason": "If the Hatter dies on or after day 5, the Demon cannot choose Leviathan."
},
{
"id": "Lil' Monsta",
"reason": "If a Demon chooses Lil' Monsta, they also choose a Minion to become and babysit Lil' Monsta tonight."
},
{
"id": "Riot",
"reason": "If the Hatter dies, Riot is in play and a Riot chooses a different Demon, a normal evil team is created from the Riot players. If the Hatter dies and the Demon chooses Riot, Minions become Riot too."
}
]
},
{
"id": "Kazali",
"hatred": [
{
"id": "Bounty Hunter",
"reason": "An evil Townsfolk is only created if the Bounty Hunter is still in play after the Kazali acts."
},
{
"id": "Choirboy",
"reason": "The Kazali can not choose the King to become a Minion if a Choirboy is in play."
},
{
"id": "Goon",
"reason": "If the Kazali chooses the Goon to become a Minion, remaining Minions choices are decided by the Storyteller."
},
{
"id": "Huntsman",
"reason": "If the Kazali chooses the Damsel to become a Minion, and a Huntsman is in play, a good player becomes the Damsel."
},
{
"id": "Marionette",
"reason": "If the Kazali chooses to create a Marionette, they must choose one of their neighbors."
}
]
}
]

File diff suppressed because it is too large Load diff

View file

@ -1,6 +1,6 @@
{
"menu":{
"grimoire":{
"menu": {
"grimoire": {
"title": "Grimoire",
"hide": "Hide",
"show": "Show",
@ -16,8 +16,8 @@
"ringBell": "Ring Bell",
"organGrinder ": "Organ Grinder Vote"
},
"session":{
"title":{
"session": {
"title": {
"player": "Playing",
"host": "Hosting",
"create": "Live Session"
@ -32,20 +32,20 @@
"voteHistory": "Vote history",
"leave": "Leave Session"
},
"players":{
"players": {
"title": "Players",
"add": "Add",
"randomize": "Randomize",
"removeAll": "Remove all"
},
"characters":{
"characters": {
"title": "Characters",
"selectEdition": "Select Edition",
"assign": "Choose & Assign",
"addFabled": "Add Fabled",
"removeAll": "Remove all"
},
"help":{
"help": {
"title": "Help",
"reference": "Reference Sheet",
"nightOrder": "Night Order Sheet",
@ -54,7 +54,7 @@
"source": "Source code"
}
},
"prompt":{
"prompt": {
"background": "Enter custom background URL",
"createSession": "Enter a channel number / name for your session",
"sendRoles": "Do you want to distribute assigned characters to all SEATED players?",
@ -69,7 +69,7 @@
"customError": "Error loading custom script",
"customNote": "Add a custom reminder node"
},
"vote":{
"vote": {
"nominates": "nominates",
"callexile": "calls for the exile of",
"exclam": "!",
@ -92,7 +92,7 @@
"seatToVote": "Pleas claim a seat to vote",
"doVote": "Go"
},
"townsquare":{
"townsquare": {
"others": "Other characters",
"bluffs": "Demon bluffs",
"fabled": "Fabled",
@ -132,11 +132,11 @@
}
}
},
"towninfo":{
"addPlayers":"Please add more players!",
"nightPhase":"Night Phase"
"towninfo": {
"addPlayers": "Please add more players!",
"nightPhase": "Night Phase"
},
"player":{
"player": {
"handUp": "Hand UP",
"handDown": "Hand DOWN",
"cancel": "Cancel",
@ -155,15 +155,15 @@
"vacateSeat": "Vacate seat",
"occupiedSeat": "Seat occupied"
},
"intro":{
"intro": {
"header": "Welcome to the (unofficial) Virtual Town Square and Grimoire for Blood on the Clocktower! Please add more players through the",
"menu": "Menu",
"body": "on the top right or by pressing [A]. You can also join a game session by pressing [J].",
"footerStart": "This project is free and open source and can be found on",
"footerEnd": ". It is not affiliated with The Pandemonium Institute. \"Blood on the Clocktower\" is a trademark of Steven Medway and The Pandemonium Institute."
},
"modal":{
"edition":{
"modal": {
"edition": {
"title": "Select an edition:",
"tab": {
"official": "Official scripts",
@ -200,20 +200,25 @@
"firstNight": "First Night",
"otherNights": "Other Nights",
"minionInfo": "Minion info",
"minionInfoDescription": "If more than one Minion, they all make eye contact with each other. Show the “This is the Demon” card. Point to the Demon.",
"minionInfoDescription": "If more than one Minion, they all make eye contact with each other. Show the “This is the Demon” card. Point to the Demon.",
"demonInfo": "Demon info & bluffs",
"demonInfoDescription": "• Show the “These are your minions” card. Point to each Minion. • Show the “These characters are not in play” card. Show 3 character tokens of good characters not in play."
"demonInfoDescription": "Show the “These are your minions” card. Point to each Minion. Show the “These characters are not in play” card. Show 3 character tokens of good characters not in play.",
"dawn": "Dawn",
"dawnDescription1": "Wake all players.",
"dawnDescription2": "Wake all players, then announce who died this night.",
"dusk": "Dusk",
"duskDescription": "End the day, and put all players to sleep."
},
"reference": {
"title": "Character Reference",
"jinxed": "Jinxed",
"notfirstnight": "*Not the first night",
"teamNames": {
"townsfolk": "townfolk",
"outsider": "outsider",
"minion": "minion",
"demon": "demon"
}
"title": "Character Reference",
"jinxed": "Jinxed",
"notfirstnight": "*Not the first night",
"teamNames": {
"townsfolk": "townfolk",
"outsider": "outsider",
"minion": "minion",
"demon": "demon"
}
},
"reminder": {
"title": "Choose a reminder token:",
@ -230,13 +235,13 @@
"roles": {
"titleStart": "Select the characters for ",
"titleEnd": " players:",
"warning":"Warning: there are characters selected that modify the game setup! The randomizer does not account for these characters.",
"warning": "Warning: there are characters selected that modify the game setup! The randomizer does not account for these characters.",
"allowMultiple": "Allow duplicate characters",
"assignStart": "Assign ",
"assignEnd": " characters randomly",
"shuffle": "Shuffle characters"
},
"voteHistory":{
"voteHistory": {
"title": "Vote history",
"accessibility": "Accessible to players",
"clear": "Clear for everyone",

View file

@ -1,19 +1,24 @@
[
{
"id": "doomsayer",
"name": "Prédicateur",
"team": "fabled",
"firstNightReminder": "",
"otherNightReminder": "",
"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."
"name": "Prédicateur",
"team": "fabled",
"firstNightReminder": "",
"otherNightReminder": "",
"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."
},
{
"id": "angel",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Protégé", "Catastrophe"],
"reminders": [
"Protégé",
"Catastrophe"
],
"setup": false,
"name": "Ange",
"team": "fabled",
@ -23,7 +28,9 @@
"id": "buddhist",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Vétéran"],
"reminders": [
"Vétéran"
],
"setup": false,
"name": "Bouddhiste",
"team": "fabled",
@ -33,7 +40,9 @@
"id": "hellslibrarian",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Catastrophe"],
"reminders": [
"Catastrophe"
],
"setup": false,
"name": "Libraire infernal",
"team": "fabled",
@ -43,7 +52,10 @@
"id": "revolutionary",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Même camp", "Utilisé"],
"reminders": [
"Même camp",
"Utilisé"
],
"setup": false,
"name": "Révolutionnaire",
"team": "fabled",
@ -62,9 +74,11 @@
{
"id": "toymaker",
"firstNightReminder": "",
"otherNight": 1,
"otherNight": 2,
"otherNightReminder": "Si le Démon pourrait terminer la partie cette nuit, mais qu'il a toujours son marqueur 'nuit sans attaque', il n'agit pas cette nuit (ne le réveillez pas)",
"reminders": ["Nuit sans attaque"],
"reminders": [
"Nuit sans attaque"
],
"setup": false,
"name": "Fabricant de Jouet",
"team": "fabled",
@ -74,7 +88,9 @@
"id": "fibbin",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Utilisé"],
"reminders": [
"Utilisé"
],
"setup": false,
"name": "Mensonge",
"team": "fabled",
@ -83,9 +99,14 @@
{
"id": "duchess",
"firstNightReminder": "",
"otherNight": 1,
"otherNight": 2,
"otherNightReminder": "Reveillez chaque visiteur dans l'ordre un par un. Indiquez à chacun d'entre eux combien de Visiteurs sont mauvais. Excepté celui qui reçoit les fausses informations qui recevra à la place n'importe quel autre nombre.",
"reminders": ["Visiteur 1", "Visiteur 2", "Visiteur 3", "Fausse Info"],
"reminders": [
"Visiteur 1",
"Visiteur 2",
"Visiteur 3",
"Fausse Info"
],
"setup": false,
"name": "Duchesse",
"team": "fabled",
@ -105,7 +126,9 @@
"id": "spiritofivory",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Pas de méchant suplémentaire"],
"reminders": [
"Pas de méchant suplémentaire"
],
"setup": false,
"name": "Esprit d'Ivoire",
"team": "fabled",
@ -127,20 +150,46 @@
"firstNight": 1,
"firstNightReminder": "Marquez un joueur comme \"Sûr\". Réveillez chaque joueur Mauvais et indiquez lui qui est ce joueur.",
"otherNightReminder": "",
"reminders": ["Sûr"],
"reminders": [
"Sûr"
],
"setup": false,
"name": "Chasseur dorages",
"team": "fabled",
"ability": "Désignez un rôle de gentil. S'il est en jeu, il ne peut mourrir que par exécution, mais les joueurs mauvais savent qui a ce rôle."
},
{
"id": "deusexfiasco",
"id": "ferryman",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": ["Whoops"],
"reminders": [],
"setup": false,
"name": "Deus ex Fiasco",
"name": "Passeur",
"team": "fabled",
"ability": "Une fois par partie, le Narrateur fera une \"Erreur\", la corrigera et l'admettra publiquement."
"ability": "Le dernier jour, tous les morts regagnent leur jeton de vote."
},
{
"id": "gardener",
"firstNightReminder": "",
"otherNightReminder": "",
"reminders": [
"Rôle choisi"
],
"setup": false,
"name": "Jardinier",
"team": "fabled",
"ability": "Le Narrateur choisit le rôle dun ou de plusieurs joueurs."
},
{
"id": "bootlegger",
"firstNight": 1,
"firstNightReminder": "Si applicables, utilisez la (les) règle(s) faite(s)-maison.",
"otherNight": 2,
"otherNightReminder": "Si applicables, utilisez la (les) règle(s) faite(s)-maison.",
"reminders": [],
"setup": false,
"name": "Contrebandier",
"team": "fabled",
"ability": "Ce script possède des rôles ou des règles faits-maison."
}
]

View file

@ -40,6 +40,10 @@
{
"id": "Politician",
"reason": "Un Chaudronnier ne peut pas créer un Politicien mauvais. "
},
{
"id": "Village Idiot",
"reason": "S'il y a moins de trois Idiots du Village, le Chaudronnier a le droit d'en créer un nouveau. L'Idiot du Village ivre peut alors changer."
}
]
},
@ -82,6 +86,10 @@
{
"id": "Mayor",
"reason": "Si le Léviathan est en jeu et qu'il n'y a pas d'exécution le cinquième jour, les bons gagnent. "
},
{
"id": "Pit-Hag",
"reason": "Après le jour 5, le Chaudronnier ne peut pas créer de Léviathan."
}
]
},
@ -397,5 +405,163 @@
"reason": "Un seul personnage maudit peut être en jeu à la fois. Les joueurs mauvais commencent la partie en sachant de quel joueur et quel rôle il s'agit."
}
]
},
{
"id": "Yaggababble",
"hatred": [
{
"id": "Exorcist",
"reason": "Si l'Exorciste désigne le Yaggablabla, la capacité du Yaggablabla ne tue personne cette nuit."
}
]
},
{
"id": "Summoner",
"hatred": [
{
"id": "Alchemist",
"reason": "Si un Alchimiste-Invocateur est en jeu, la partie commence avec un Démon, comme d'habitude. Si l'Alchimiste-Invocateur désigne un joueur, il le transforme en Démon mais ne change pas son alignement."
},
{
"id": "Clockmaker",
"reason": "Si l'Invocateur est en jeu, l'Horloger ne reçoit aucune info jusqu'à ce qu'un Démon soit créé."
},
{
"id": "Poppy Grower",
"reason": "Si le Planteur de pavot est en vie lorsque l'Invocateur agit, l'Invocateur choisit un rôle de Démon, mais c'est le Narrateur qui choisit un joueur."
},
{
"id": "Marionette",
"reason": "La Marionnette est voisine de l'Invocateur. L'Invocateur sait qui est la Marionnette."
},
{
"id": "Kazali",
"reason": "Le Kazali ne peut pas créer un Invocateur."
},
{
"id": "Legion",
"reason": "Si l'Invocateur crée une Légion, la majorité des joueurs (y compris les joueurs mauvais) deviennent des Légions mauvaises."
},
{
"id": "Riot",
"reason": "Si l'Invocateur crée une Émeute, le joueur choisi et tous les mauvais deviennent des Émeutes. Le joueur choisi doit être un des bons voisins vivants de l'Invocateur."
}
]
},
{
"id": "Vizier",
"hatred": [
{
"id": "Magician",
"reason": "Un seul personnage maudit peut être en jeu à la fois. Les joueurs mauvais commencent la partie en sachant de quel joueur et quel rôle il s'agit."
},
{
"id": "Alchemist",
"reason": "Si l'Alchimiste a la capacité du Vizir, il peut exécuter si et seulement si au moins 3 joueurs votent, peu importe leur alignement."
},
{
"id": "Courtier",
"reason": "Si le Vizir perd sa capacité, il l'apprend. Si le Vizir est exécuté alors qu'il a sa capacité, son équipe gagne."
},
{
"id": "Preacher",
"reason": "Si le Vizir perd sa capacité, il l'apprend. Si le Vizir est exécuté alors qu'il a sa capacité, son équipe gagne."
},
{
"id": "Investigator",
"reason": "Si l'Enquêteur apprend qu'un Vizir est en jeu, la présence du Vizir n'est pas annoncée publiquement."
},
{
"id": "Fearmonger",
"reason": "Le Vizir apprend quel joueur a été désigné par le Semeur de peur, et ne peut pas forcer l'exécution de ce joueur."
},
{
"id": "Lil' Monsta",
"reason": "Le Vizir peut mourir par exécution s'il a la garde du Bébé Monstre."
}
]
},
{
"id": "Hatter",
"hatred": [
{
"id": "Legion",
"reason": "Si le Chapelier meurt alors qu'une Légion est en jeu, rien ne se passe. Si le Chapelier meurt et qu'un mauvais choisit de devenir Légion, tous les joueurs mauvais deviennent une Légion."
},
{
"id": "Leviathan",
"reason": "Si le Chapelier est mort durant ou après le cinquième jour, le Démon ne peut pas choisir de devenir Léviathan."
},
{
"id": "Lil' Monsta",
"reason": "Si un Démon choisit de devenir Bébé monstre, il choisit aussi un rôle de Serviteur et devient baby-sitter pour cette nuit."
},
{
"id": "Riot",
"reason": "Si à la mort du Chapelier, une Émeute choisit de devenir un autre Démon, les autres Émeutes doivent devenir des Serviteurs. Si à la mort du Chapelier, le Démon choisit de devenir Émeute, les Serviteurs deviennent des Émeutes aussi."
}
]
},
{
"id": "Kazali",
"hatred": [
{
"id": "Bounty Hunter",
"reason": "Un Villageois ne devient mauvais que si le Mercenaire est toujours en jeu après l'action du Kazali."
},
{
"id": "Choirboy",
"reason": "Le Kazali ne peut pas choisir de changer le Roi en Serviteur si l'Enfant de choeur est en jeu."
},
{
"id": "Goon",
"reason": "Si le Kazali choisit de transformer l'Homme de main en Serviteur, il devient ivre, et les Serviteurs restants sont choisis par le Narrateur sans prendre en compte les décisions du Kazali."
},
{
"id": "Huntsman",
"reason": "Si le Kazali transforme la Demoiselle en Serviteur, et si le Chasseur est en jeu, un joueur bon devient Demoiselle."
},
{
"id": "Marionette",
"reason": "Si le Kazali veut créer une Marionnette, il doit choisir l'un de ses voisins."
}
]
},
{
"id": "Plague Doctor",
"hatred": [
{
"id": "Baron",
"reason": "Si le Narrateur gagne la capacité du Baron, jusqu'à deux joueurs deviennent des Étrangers qui ne sont pas en jeu."
},
{
"id": "Boomdandy",
"reason": "Si le Médecin de Peste est exécuté et que le Narrateur aurait du gagner une capacité d'Homme-bombe, la capacité d'Homme-bombe s'active immédiatement."
},
{
"id": "Evil Twin",
"reason": "Le Narrateur ne peut pas gagner une capacité de Jumeau maléfique si le Médecin de Peste meurt."
},
{
"id": "Fearmonger",
"reason": "Si le Médecin de Peste aurait du donner la capacité de Semeur de peur au Narrateur, un Serviteur en vie gagne la capacité de Semeur de peur en plus de sa propre capacité, et l'apprend."
},
{
"id": "Goblin",
"reason": "Si le Médecin de Peste aurait du donner la capacité de Gobelin au Narrateur, un Serviteur en vie gagne la capacité de Gobelin en plus de sa propre capacité, et l'apprend."
},
{
"id": "Scarlet Woman",
"reason": "Si le Médecin de Peste aurait du donner la capacité de Gourgandine au Narrateur, un Serviteur en vie gagne la capacité de Gourgandine en plus de sa propre capacité, et l'apprend."
},
{
"id": "Spy",
"reason": "Si le Médecin de Peste aurait du donner la capacité d'Espion au Narrateur, un Serviteur en vie gagne la capacité d'Espion en plus de sa propre capacité, et l'apprend."
},
{
"id": "Marionette",
"reason": "Si l'un des voisins du Démon est un Villageois ou Étranger et est vivant quand le Médecin de Peste meurt, ce joueur devient une mauvaise Marionnette. S'il y a déjà un joueur mauvais de plus, cela n'arrive pas."
}
]
}
]

File diff suppressed because it is too large Load diff

View file

@ -1,6 +1,6 @@
{
"menu":{
"grimoire":{
"menu": {
"grimoire": {
"title": "Grimoire",
"hide": "Cacher",
"show": "Montrer",
@ -16,8 +16,8 @@
"ringBell": "Sonner Clocher",
"organGrinder": "Vote d'Organiste"
},
"session":{
"title":{
"session": {
"title": {
"player": "Joueur",
"host": "Hôte",
"create": "Session Live"
@ -32,20 +32,20 @@
"voteHistory": "Historique Votes",
"leave": "Quitter Session"
},
"players":{
"players": {
"title": "Joueurs",
"add": "Ajouter",
"randomize": "Mélanger Sièges",
"removeAll": "Retirer Joueurs"
},
"characters":{
"characters": {
"title": "Personnages",
"selectEdition": "Choisir Scénario",
"assign": "Attribuer Rôles",
"addFabled": "Ajouter Fabuleux",
"removeAll": "Effacer Rôles"
},
"help":{
"help": {
"title": "Aide",
"reference": "Référence rôles",
"nightOrder": "Ordre Nocturne",
@ -54,7 +54,7 @@
"source": "Code Source"
}
},
"prompt":{
"prompt": {
"background": "Entrez l'URL de l'image de fond",
"createSession": "Entrez un nom ou numéro de session",
"sendRoles": "Voulez-vous envoyer les rôles à tous les joueurs ASSIS ?",
@ -69,7 +69,7 @@
"customError": "Erreur lors du chargement du script",
"customNote": "Ajouter une note personnalisée"
},
"vote":{
"vote": {
"nominates": "accuse",
"callexile": "veut exiler",
"exclam": " !",
@ -92,7 +92,7 @@
"seatToVote": "Asseyez-vous pour pouvoir voter",
"doVote": "Votez"
},
"townsquare":{
"townsquare": {
"others": "Autres Rôles",
"bluffs": "Bluffs de Démon",
"fabled": "Fabuleux",
@ -132,11 +132,11 @@
}
}
},
"towninfo":{
"towninfo": {
"addPlayers": "Appuyez sur [A] pour ajouter plus de joueurs !",
"nightPhase": "C'est la nuit"
},
"player":{
"player": {
"handUp": "Main levée",
"handDown": "Main baissée",
"cancel": "Annuler",
@ -155,15 +155,15 @@
"vacateSeat": "Libérer le Siège",
"occupiedSeat": "Siège Occupé"
},
"intro":{
"intro": {
"header": "Bienvenue sur le Centre-ville Virtuel (non-officiel) pour Blood on the Clocktower! Veuillez ajouter des Joueurs via le",
"menu": "Menu",
"body": "en haut à droite ou en appuyant sur [A] pour commencer. Vous pouvez aussi rejoindre une session en appuyant sur [J].",
"footerStart": "Ce programme est libre et ses sources peuvent être trouvées sur",
"footerEnd": ". Ce site n'est pas affilié à The Pandemonium Institute. \"Blood on the Clocktower\" est une marque déposée de Steven Medway & The Pandemonium Institute."
},
"modal":{
"edition":{
"modal": {
"edition": {
"title": "Choisir un Scénario :",
"tab": {
"official": "Scénarios officiels",
@ -200,20 +200,25 @@
"firstNight": "Première Nuit",
"otherNights": "Autres Nuits",
"minionInfo": "Informations Serviteurs",
"minionInfoDescription": "S'il y a plusieurs Serviteurs, ils apprennent qui sont les autres Serviteurs. Indiquez aux Serviteurs qui est le Démon.",
"minionInfoDescription": "S'il y a plusieurs Serviteurs, ils apprennent qui sont les autres Serviteurs. Indiquez aux Serviteurs qui est le Démon.",
"demonInfo": "Info & Bluffs Démon",
"demonInfoDescription": "• Indiquez au Démon qui sont ses serviteurs.• Indiquez les rôles de 3 personnages Bons qui ne sont pas en jeu."
"demonInfoDescription": "Indiquez au Démon qui sont ses serviteurs. Indiquez les rôles de 3 personnages bons qui ne sont pas en jeu.",
"dawn": "Matin",
"dawnDescription1": "Réveillez les joueurs.",
"dawnDescription2": "Réveillez les joueurs, puis annoncez qui est mort cette nuit",
"dusk": "Tombée de la nuit",
"duskDescription": "Terminez la journée, et endormez les joueurs."
},
"reference": {
"title": "Réference de rôles",
"jinxed": "Jinx",
"notfirstnight": "* Pas la première nuit",
"teamNames": {
"townsfolk": "villageois",
"outsider": "étranger",
"minion": "serviteur",
"demon": "démon"
}
"title": "Réference de rôles",
"jinxed": "Jinx",
"notfirstnight": "* Pas la première nuit",
"teamNames": {
"townsfolk": "villageois",
"outsider": "étranger",
"minion": "serviteur",
"demon": "démon"
}
},
"reminder": {
"title": "Apposer une note:",
@ -230,13 +235,13 @@
"roles": {
"titleStart": "Selectionner les personnages pour ",
"titleEnd": " joueurs:",
"warning":"Attention: certains des personnages sélectionnés changent la distribution de début de partie ! La distribution aléatoire n'effectue pas elle-même ces changements, pensez à modifier les attributions en conséquence avant d'envoyer les rôles aux joueurs.",
"warning": "Attention: certains des personnages sélectionnés changent la distribution de début de partie ! La distribution aléatoire n'effectue pas elle-même ces changements, pensez à modifier les attributions en conséquence avant d'envoyer les rôles aux joueurs.",
"allowMultiple": "Permettre les doublons de personnages",
"assignStart": "Attribuer aléatoirement ces ",
"assignEnd": " rôles",
"shuffle": "Tirer les personnages au sort"
},
"voteHistory":{
"voteHistory": {
"title": "Historique de votes",
"accessibility": "Accessible aux Joueurs",
"clear": "Effacer pour tous",

View file

@ -28,33 +28,33 @@ const getters = {
nightOrder({ players, fabled }) {
const firstNight = [0];
const otherNight = [0];
players.forEach(({ role }) => {
if (role.firstNight && !firstNight.includes(role.firstNight)) {
firstNight.push(role.firstNight);
}
if (role.otherNight && !otherNight.includes(role.otherNight)) {
otherNight.push(role.otherNight);
}
});
fabled.forEach((role) => {
if (role.firstNight && !firstNight.includes(role.firstNight)) {
firstNight.push(role.firstNight);
if (role.firstNight && !firstNight.includes(role)) {
firstNight.push(role);
}
if (role.otherNight && !otherNight.includes(role.otherNight)) {
otherNight.push(role.otherNight);
if (role.otherNight && !otherNight.includes(role)) {
otherNight.push(role);
}
});
firstNight.sort((a, b) => a - b);
otherNight.sort((a, b) => a - b);
players.forEach(({ role }) => {
if (role.firstNight && !firstNight.includes(role)) {
firstNight.push(role);
}
if (role.otherNight && !otherNight.includes(role)) {
otherNight.push(role);
}
});
firstNight.sort((a, b) => a.firstNight - b.firstNight);
otherNight.sort((a, b) => a.otherNight - b.otherNight);
const nightOrder = new Map();
players.forEach((player) => {
const first = Math.max(firstNight.indexOf(player.role.firstNight), 0);
const other = Math.max(otherNight.indexOf(player.role.otherNight), 0);
const first = Math.max(firstNight.indexOf(player.role), 0);
const other = Math.max(otherNight.indexOf(player.role), 0);
nightOrder.set(player, { first, other });
});
fabled.forEach((role) => {
const first = Math.max(firstNight.indexOf(role.firstNight), 0);
const other = Math.max(otherNight.indexOf(role.otherNight), 0);
const first = Math.max(firstNight.indexOf(role), 0);
const other = Math.max(otherNight.indexOf(role), 0);
nightOrder.set(role, { first, other });
});
return nightOrder;
@ -86,7 +86,6 @@ const actions = {
id,
pronouns,
}));
commit("setFabled", { fabled: [] });
}
commit("set", players);
commit("setBluff");

View file

@ -4,3 +4,4 @@ $outsider: #46d5ff;
$minion: #ff6900;
$demon: #ce0100;
$traveler: #cc04ff;
$default: #4E4E4E;