2022-01-09 09:33:47 +01:00
|
|
|
/* -------------------------------------------- */
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
const dureeGaldrText = { "1d5a": "Actions", "1d10t": "Tours", "1d10m": "Minutes", "1d10h": "Heures", "1d5j": "Jours"};
|
|
|
|
const ciblesGaldrText = { "1": "1", "2_4": "2 à 4", "5_9": "5 à 9", "10_49": "10 à 49", "50plus": "50 et plus"};
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
export class YggdrasillUtility {
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static async preloadHandlebarsTemplates() {
|
|
|
|
|
|
|
|
const templatePaths = [
|
|
|
|
'systems/fvtt-yggdrasill/templates/actor-sheet.html',
|
2024-05-01 09:33:34 +02:00
|
|
|
'systems/fvtt-yggdrasill/templates/editor-notes-gm.html'
|
2022-01-09 09:33:47 +01:00
|
|
|
]
|
|
|
|
return loadTemplates(templatePaths);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static createDirectSortedOptionList( min, max) {
|
|
|
|
let options = [];
|
|
|
|
for(let i=min; i<=max; i++) {
|
|
|
|
options.push( {value:i, text: `${i}` } );
|
|
|
|
}
|
|
|
|
return options;
|
|
|
|
}
|
2024-05-01 09:33:34 +02:00
|
|
|
/* -------------------------------------------- */
|
|
|
|
static createOptions( min, max) {
|
|
|
|
let options = [];
|
|
|
|
for(let i=min; i<=max; i++) {
|
|
|
|
options.push( {key:i, label: `${i}` } );
|
|
|
|
}
|
|
|
|
return options;
|
|
|
|
}
|
2022-01-09 09:33:47 +01:00
|
|
|
/* -------------------------------------------- */
|
|
|
|
static createDirectOptionList( min, max) {
|
|
|
|
let options = {};
|
|
|
|
for(let i=min; i<=max; i++) {
|
|
|
|
options[`${i}`] = `${i}`;
|
|
|
|
}
|
|
|
|
return options;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static buildListOptions(min, max) {
|
|
|
|
let options = ""
|
|
|
|
for (let i = min; i <= max; i++) {
|
|
|
|
options += `<option value="${i}">${i}</option>`
|
|
|
|
}
|
|
|
|
return options;
|
|
|
|
}
|
|
|
|
/* -------------------------------------------- */
|
2024-05-01 09:33:34 +02:00
|
|
|
static buildListOptions(min, max) {
|
2022-01-09 09:33:47 +01:00
|
|
|
let options = ""
|
2024-05-01 09:33:34 +02:00
|
|
|
for (let i = min; i <= max; i++) {
|
|
|
|
options += `<option value="${i}">${i}</option>`
|
|
|
|
}
|
2022-01-09 09:33:47 +01:00
|
|
|
return options;
|
|
|
|
}
|
2024-05-01 09:33:34 +02:00
|
|
|
|
2022-01-09 09:33:47 +01:00
|
|
|
/* -------------------------------------------- */
|
|
|
|
static onSocketMesssage( msg ) {
|
|
|
|
if( !game.user.isGM ) return; // Only GM
|
|
|
|
|
|
|
|
if (msg.name == 'msg_attack' ) {
|
|
|
|
this.performAttack( msg.data );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static async loadCompendiumData(compendium) {
|
|
|
|
const pack = game.packs.get(compendium);
|
|
|
|
return await pack?.getDocuments() ?? [];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static async loadCompendium(compendium, filter = item => true) {
|
|
|
|
let compendiumData = await YggdrasillUtility.loadCompendiumData(compendium);
|
|
|
|
return compendiumData.filter(filter);
|
|
|
|
}
|
|
|
|
|
2022-01-23 00:07:47 +01:00
|
|
|
/* -------------------------------------------- */
|
2023-05-25 15:13:41 +02:00
|
|
|
static async specificYggRoll( nbDice, isFurorUsage = false) {
|
2022-01-23 00:07:47 +01:00
|
|
|
let rawDices = []
|
|
|
|
let rolls = []
|
|
|
|
let maxTab = []
|
2023-02-13 14:38:20 +01:00
|
|
|
let maxTabMaxIndex = isFurorUsage ? nbDice : 2;
|
2022-01-23 00:07:47 +01:00
|
|
|
|
|
|
|
for (let i=0; i<nbDice; i++) {
|
2024-05-01 09:33:34 +02:00
|
|
|
rolls[i] = await new Roll("1d10x10").roll( ) //+sumDice+"+"+rollData.furorUsage+"d10+"+niveauCompetence+"+"+rollData.finalBM).roll( { async: false} );
|
2022-01-23 00:07:47 +01:00
|
|
|
if ( i == nbDice-1 ) {
|
|
|
|
await this.showDiceSoNice(rolls[i], game.settings.get("core", "rollMode") );
|
|
|
|
} else {
|
|
|
|
this.showDiceSoNice(rolls[i], game.settings.get("core", "rollMode") );
|
|
|
|
}
|
|
|
|
rawDices.push({ 'result': rolls[i].total});
|
2023-02-13 14:38:20 +01:00
|
|
|
}
|
2022-01-23 00:07:47 +01:00
|
|
|
|
2023-02-13 14:38:20 +01:00
|
|
|
rolls.sort((a,b) => a.total-b.total);
|
|
|
|
rolls.reverse();
|
|
|
|
|
|
|
|
for (let i=0; i<maxTabMaxIndex; i++) {
|
|
|
|
maxTab[i] = {idx: 0, value: 0};
|
|
|
|
if (rolls[i].total != undefined) maxTab[i].value = rolls[i].total;
|
2022-01-23 00:07:47 +01:00
|
|
|
}
|
2023-02-13 14:38:20 +01:00
|
|
|
|
|
|
|
return { rawDices: rawDices, maxTab: maxTab, rolls: rolls }
|
2022-01-23 00:07:47 +01:00
|
|
|
}
|
|
|
|
|
2022-01-09 09:33:47 +01:00
|
|
|
/* -------------------------------------------- */
|
|
|
|
static async rollAttribute( rollData ) {
|
|
|
|
// Init stuff
|
|
|
|
let isCritical = false;
|
|
|
|
let isFailure = false;
|
|
|
|
let isSuccess = false;
|
|
|
|
let marge = 0;
|
|
|
|
let niveau = rollData.subAttr.value;
|
|
|
|
|
|
|
|
// Bonus/Malus total
|
2024-05-01 09:33:34 +02:00
|
|
|
rollData.finalBM = Number(rollData.bonusMalus);
|
2022-01-09 09:33:47 +01:00
|
|
|
// Gestion cas blessé (malus de -3)
|
|
|
|
if ( rollData.isBlesse) { // Cas blesse : malus de -3
|
|
|
|
rollData.finalBM -= 3;
|
|
|
|
}
|
|
|
|
|
2022-01-23 00:07:47 +01:00
|
|
|
let results = await this.specificYggRoll( 2 )
|
|
|
|
rollData.rawDices = results.rawDices
|
|
|
|
rollData.maxTab = results.maxTab
|
|
|
|
rollData.rolls = results.rolls
|
2024-05-01 09:33:34 +02:00
|
|
|
rollData.bonus = niveau + Number(rollData.finalBM)
|
2022-01-12 11:32:08 +01:00
|
|
|
|
2024-05-01 09:33:34 +02:00
|
|
|
rollData.finalTotal = Number(rollData.maxTab[0].value) + Number(rollData.maxTab[1].value);
|
|
|
|
rollData.finalTotal += Number(rollData.bonus)
|
2022-01-23 00:07:47 +01:00
|
|
|
|
2022-01-09 09:33:47 +01:00
|
|
|
// Compute total SR
|
2024-05-01 09:33:34 +02:00
|
|
|
rollData.srFinal = Number(rollData.sr);
|
2022-01-23 00:07:47 +01:00
|
|
|
if ( rollData.bonusdefense ) {
|
2024-05-01 09:33:34 +02:00
|
|
|
rollData.srFinal += Number(rollData.bonusdefense);
|
2022-01-23 00:07:47 +01:00
|
|
|
}
|
2022-01-09 09:33:47 +01:00
|
|
|
if ( rollData.srFinal > 0 ) {
|
2022-01-12 11:32:08 +01:00
|
|
|
isCritical = rollData.finalTotal >= rollData.srFinal*2;
|
|
|
|
isSuccess = rollData.finalTotal >= rollData.srFinal;
|
|
|
|
marge = rollData.finalTotal - rollData.srFinal;
|
2022-01-09 09:33:47 +01:00
|
|
|
}
|
|
|
|
|
2022-01-23 00:07:47 +01:00
|
|
|
if (rollData.rolls[0].dice[0].results[0].result == 1 && rollData.rolls[1].dice[0].results[0].result == 1) {
|
2022-01-09 09:33:47 +01:00
|
|
|
isFailure = true;
|
|
|
|
}
|
|
|
|
// Dégats
|
|
|
|
if ( isSuccess && rollData.subAttr.degats ) {
|
|
|
|
rollData.degatsExplain = `Marge(${marge}) + Physique(${rollData.valuePhysique}) + 1d10`;
|
2024-05-01 09:33:34 +02:00
|
|
|
rollData.rollDegats = await new Roll("1d10+"+marge+"+"+rollData.valuePhysique).roll( );
|
2022-01-09 09:33:47 +01:00
|
|
|
await this.showDiceSoNice(rollData.rollDegats, game.settings.get("core", "rollMode") );
|
|
|
|
rollData.degats = rollData.rollDegats.total;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stockage resultats
|
|
|
|
rollData.isFailure = isFailure;
|
|
|
|
rollData.isSuccess = isSuccess;
|
|
|
|
rollData.isCritical = isCritical;
|
|
|
|
rollData.marge = marge;
|
|
|
|
|
|
|
|
console.log("ROLLLL ATTR!!!!", rollData);
|
|
|
|
|
|
|
|
this.createChatWithRollMode( rollData.alias, {
|
|
|
|
content: await renderTemplate(`systems/fvtt-yggdrasill/templates/chat-generic-result.html`, rollData)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static async rollYggdrasill( rollData ) {
|
2022-01-16 10:09:16 +01:00
|
|
|
let sumDice = ( rollData.isEpuise | rollData.isMeurtri) ? 1 : 2;
|
2022-01-09 09:33:47 +01:00
|
|
|
|
|
|
|
// Init stuff
|
|
|
|
let isCritical = false;
|
|
|
|
let isFailure = false;
|
|
|
|
let isSuccess = false;
|
|
|
|
let marge = 0;
|
|
|
|
let nbDice = rollData.selectedCarac.value;
|
|
|
|
let niveauCompetence = 0;
|
|
|
|
|
|
|
|
// Select niveau de competence/arme/carac
|
|
|
|
if ( rollData.mode != "carac" ) {
|
2022-09-21 11:42:09 +02:00
|
|
|
niveauCompetence = rollData.competence.system.niveau;
|
2022-01-23 22:01:13 +01:00
|
|
|
} else {
|
|
|
|
niveauCompetence = rollData.selectedCarac.value;
|
|
|
|
}
|
2022-01-09 09:33:47 +01:00
|
|
|
|
|
|
|
// Bonus/Malus total
|
2024-05-01 09:33:34 +02:00
|
|
|
rollData.finalBM = Number(rollData.bonusMalus);
|
2022-01-09 09:33:47 +01:00
|
|
|
if ( rollData.attackDef) {
|
|
|
|
rollData.finalBM -= rollData.attackDef.malus;
|
|
|
|
}
|
2022-09-21 11:42:09 +02:00
|
|
|
if ( rollData.sort && rollData.sort.system.malus ) {
|
|
|
|
rollData.finalBM += rollData.sort.system.malus;
|
2022-01-09 09:33:47 +01:00
|
|
|
}
|
|
|
|
// Gestion cas blessé (malus de -3)
|
|
|
|
if ( rollData.isBlesse) { // Cas blesse : malus de -3
|
|
|
|
rollData.finalBM -= 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sumDice > nbDice) sumDice = nbDice;
|
2022-01-23 00:07:47 +01:00
|
|
|
let results = await this.specificYggRoll( nbDice )
|
|
|
|
rollData.rawDices = results.rawDices
|
|
|
|
rollData.maxTab = results.maxTab
|
|
|
|
rollData.rolls = results.rolls
|
|
|
|
console.log("RES", results, nbDice, sumDice)
|
2022-01-11 18:19:13 +01:00
|
|
|
|
2022-01-23 00:07:47 +01:00
|
|
|
if ( rollData.furorUsage > 0 ) {
|
2023-02-13 14:38:20 +01:00
|
|
|
results = await this.specificYggRoll( rollData.furorUsage, true )
|
2022-01-23 00:07:47 +01:00
|
|
|
rollData.furorRawDices = results.rawDices
|
|
|
|
rollData.furorMaxTab = results.maxTab
|
|
|
|
rollData.furorRolls = results.rolls
|
|
|
|
let actor = game.actors.get(rollData.actorId);
|
|
|
|
actor.decrementFuror( rollData.furorUsage);
|
|
|
|
}
|
|
|
|
|
|
|
|
rollData.bonusTotal = niveauCompetence + rollData.finalBM
|
|
|
|
rollData.finalTotal = (sumDice ==1) ? rollData.maxTab[0].value : rollData.maxTab[0].value + rollData.maxTab[1].value;
|
|
|
|
rollData.furorResult = 0
|
|
|
|
for (let i=0; i<rollData.furorUsage; i++) {
|
|
|
|
rollData.furorResult += rollData.furorMaxTab[i].value
|
2022-01-11 18:19:13 +01:00
|
|
|
}
|
2024-05-01 09:33:34 +02:00
|
|
|
rollData.finalTotal += Number(rollData.furorResult) + Number(rollData.bonusTotal);
|
2022-01-23 00:07:47 +01:00
|
|
|
rollData.niveauCompetence = niveauCompetence
|
2022-01-11 18:19:13 +01:00
|
|
|
|
2022-01-09 09:33:47 +01:00
|
|
|
// Compute total SR
|
|
|
|
rollData.srFinal = rollData.sr;
|
|
|
|
if ( rollData.bonusdefense ) {
|
|
|
|
rollData.srFinal += rollData.bonusdefense;
|
2022-01-11 18:19:13 +01:00
|
|
|
}
|
|
|
|
|
2022-01-09 09:33:47 +01:00
|
|
|
if ( rollData.srFinal > 0 ) {
|
2022-01-12 09:13:47 +01:00
|
|
|
isCritical = rollData.finalTotal >= rollData.srFinal*2;
|
|
|
|
isSuccess = rollData.finalTotal >= rollData.srFinal;
|
|
|
|
marge = rollData.finalTotal - rollData.srFinal;
|
2022-01-09 09:33:47 +01:00
|
|
|
}
|
|
|
|
|
2022-01-23 00:07:47 +01:00
|
|
|
if (nbDice == 1 && rollData.rolls[0].dice[0].results[0].result == 1) {
|
2022-01-09 09:33:47 +01:00
|
|
|
isFailure = true;
|
|
|
|
}
|
2022-01-23 00:07:47 +01:00
|
|
|
if (nbDice == 2 && rollData.rolls[0].dice[0].results[0].result == 1 && rollData.rolls[1].dice[0].results[0].result == 1) {
|
2022-01-09 09:33:47 +01:00
|
|
|
isFailure = true;
|
|
|
|
}
|
|
|
|
if (nbDice >= 3 ) {
|
2022-01-11 18:19:13 +01:00
|
|
|
let nbOnes = 0
|
2022-01-23 00:07:47 +01:00
|
|
|
for (let roll of rollData.rolls) {
|
2022-01-11 18:19:13 +01:00
|
|
|
if (roll.dice[0].results[0].result == 1 ) nbOnes++;
|
|
|
|
}
|
|
|
|
isFailure = nbOnes >= 3;
|
2022-01-09 09:33:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Dégats
|
|
|
|
if ( isSuccess && (rollData.mode == "armecc" || rollData.mode == "armedist") ) {
|
2022-09-21 11:42:09 +02:00
|
|
|
rollData.degatsExplain = `Marge(${marge}) + Degats Arme(${rollData.arme.system.degat}) + Bonus Attaque(${rollData.attackDef.bonusdegats})`;
|
|
|
|
rollData.degats = marge + rollData.arme.system.degat + rollData.attackDef.bonusdegats;
|
2022-01-09 09:33:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Stockage resultats
|
|
|
|
rollData.sumDice = sumDice;
|
|
|
|
rollData.isFailure = isFailure;
|
|
|
|
rollData.isSuccess = isSuccess;
|
|
|
|
rollData.isCritical = isCritical;
|
|
|
|
rollData.marge = marge;
|
|
|
|
|
|
|
|
// Specific GALDR
|
|
|
|
if ( rollData.sort?.type == "sortgaldr" && rollData.isSuccess) {
|
2024-05-01 09:33:34 +02:00
|
|
|
let galdrRoll = await new Roll( rollData.dureeGaldr.substring(0, rollData.dureeGaldr.length - 1) ).roll( );
|
2022-01-09 09:33:47 +01:00
|
|
|
await this.showDiceSoNice(galdrRoll, game.settings.get("core", "rollMode") );
|
|
|
|
rollData.dureeGaldrText = galdrRoll.total + " " + dureeGaldrText[rollData.dureeGaldr];
|
2022-09-21 11:42:09 +02:00
|
|
|
if ( rollData.sort.system.voie == "illusion") {
|
2022-01-09 09:33:47 +01:00
|
|
|
let volume = rollData.zoneGaldr.substring(3, rollData.zoneGaldr.length);
|
|
|
|
rollData.zoneGaldrText = rollData.instinctCarac.value + " x " + volume;
|
|
|
|
} else {
|
|
|
|
rollData.ciblesGaldrText = ciblesGaldrText[rollData.nbCibles];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log("ROLLLL!!!!", rollData);
|
|
|
|
|
|
|
|
this.createChatWithRollMode( rollData.alias, {
|
|
|
|
content: await renderTemplate(`systems/fvtt-yggdrasill/templates/chat-generic-result.html`, rollData)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static getUsers(filter) {
|
2022-09-21 11:42:09 +02:00
|
|
|
return game.users.filter(filter).map(user => user.system._id);
|
2022-01-09 09:33:47 +01:00
|
|
|
}
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static getWhisperRecipients(rollMode, name) {
|
|
|
|
switch (rollMode) {
|
|
|
|
case "blindroll": return this.getUsers(user => user.isGM);
|
|
|
|
case "gmroll": return this.getWhisperRecipientsAndGMs(name);
|
|
|
|
case "selfroll": return [game.user.id];
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static getWhisperRecipientsAndGMs(name) {
|
|
|
|
return ChatMessage.getWhisperRecipients(name)
|
|
|
|
.concat(ChatMessage.getWhisperRecipients('GM'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static blindMessageToGM(chatOptions) {
|
2024-05-01 09:33:34 +02:00
|
|
|
let chatGM = foundry.utils.duplicate(chatOptions);
|
2022-01-09 09:33:47 +01:00
|
|
|
chatGM.whisper = this.getUsers(user => user.isGM);
|
|
|
|
chatGM.content = "Message aveugle de " + game.user.name + "<br>" + chatOptions.content;
|
|
|
|
console.log("blindMessageToGM", chatGM);
|
|
|
|
game.socket.emit("system.foundryvtt-yggdrasill", { msg: "msg_gm_chat_message", data: chatGM });
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static createChatMessage(name, rollMode, chatOptions) {
|
|
|
|
switch (rollMode) {
|
|
|
|
case "blindroll": // GM only
|
|
|
|
if (!game.user.isGM) {
|
|
|
|
this.blindMessageToGM(chatOptions);
|
|
|
|
|
|
|
|
chatOptions.whisper = [game.user.id];
|
|
|
|
chatOptions.content = "Message envoyé en aveugle au Gardien";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
chatOptions.whisper = this.getUsers(user => user.isGM);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
chatOptions.whisper = this.getWhisperRecipients(rollMode, name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
chatOptions.alias = chatOptions.alias || name;
|
|
|
|
ChatMessage.create(chatOptions);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static createChatWithRollMode(name, chatOptions) {
|
|
|
|
this.createChatMessage(name, game.settings.get("core", "rollMode"), chatOptions);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static async confirmDelete(actorSheet, li) {
|
|
|
|
let itemId = li.data("item-id");
|
|
|
|
let msgTxt = "<p>Etes vous certain de souhaiter supprimer cet item ?";
|
|
|
|
let buttons = {
|
|
|
|
delete: {
|
|
|
|
icon: '<i class="fas fa-check"></i>',
|
|
|
|
label: "Oui, à supprimer",
|
|
|
|
callback: () => {
|
|
|
|
actorSheet.actor.deleteEmbeddedDocuments( "Item", [itemId] );
|
|
|
|
li.slideUp(200, () => actorSheet.render(false));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
cancel: {
|
|
|
|
icon: '<i class="fas fa-times"></i>',
|
|
|
|
label: "Annuler"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
msgTxt += "</p>";
|
|
|
|
let d = new Dialog({
|
|
|
|
title: "Confirmer la suppression",
|
|
|
|
content: msgTxt,
|
|
|
|
buttons: buttons,
|
|
|
|
default: "cancel"
|
|
|
|
});
|
|
|
|
d.render(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
static async showDiceSoNice(roll, rollMode) {
|
|
|
|
if (game.modules.get("dice-so-nice")?.active) {
|
|
|
|
if (game.dice3d) {
|
|
|
|
let whisper = null;
|
|
|
|
let blind = false;
|
|
|
|
rollMode = rollMode ?? game.settings.get("core", "rollMode");
|
|
|
|
switch (rollMode) {
|
|
|
|
case "blindroll": //GM only
|
|
|
|
blind = true;
|
|
|
|
case "gmroll": //GM + rolling player
|
|
|
|
whisper = this.getUsers(user => user.isGM);
|
|
|
|
break;
|
|
|
|
case "roll": //everybody
|
|
|
|
whisper = this.getUsers(user => user.active);
|
|
|
|
break;
|
|
|
|
case "selfroll":
|
|
|
|
whisper = [game.user.id];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
await game.dice3d.showForRoll(roll, game.user, true, whisper, blind);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|