2 Commits

Author SHA1 Message Date
6c6c473147 Fixes regarding shields usage and spells
All checks were successful
Release Creation / build (release) Successful in 52s
2026-04-14 21:31:17 +02:00
2e2a917a45 Fixes regarding shields usage and spells 2026-04-14 21:31:03 +02:00
30 changed files with 242 additions and 104 deletions

View File

@@ -723,6 +723,9 @@
},
"catalyst": {
"label": "Catalyst"
},
"damageDice": {
"label": "Damage dice"
}
}
},

View File

@@ -346,6 +346,36 @@ Hooks.on(hookName, (message, html, data) => {
return
}
// Pour les sorts, rouler les dés de dégâts avec option bypass DR
if (damageType === "spell" && damageFormula) {
const bypassArmor = await foundry.applications.api.DialogV2.confirm({
window: { title: "Spell Damage" },
classes: ["lethalfantasy"],
content: "<p>Does this spell's damage bypass armor DR?</p>",
yes: { label: "Yes (ignore armor)", icon: "fa-solid fa-wand-magic-sparkles" },
no: { label: "No (apply armor DR)", icon: "fa-solid fa-shield" }
})
const rollOpts = {
type: "spell-damage",
rollType: "spell-damage",
rollName: damageFormula,
isDamage: true,
rollData: { isDamage: true },
bypassArmor: bypassArmor ?? false,
defenderId,
defenderTokenId,
actorId: actor.id,
actorName: actor.name,
actorImage: actor.img
}
const roll = new LethalFantasyRoll(damageFormula, {}, rollOpts)
await roll.evaluate()
roll.options.rollTotal = roll.total
if (game?.dice3d) await game.dice3d.showForRoll(roll, game.user, true)
await roll.toMessage()
return
}
// Pour les boutons de résultat de combat (monster damage)
if (damageType === "monster" && attackKey) {
await actor.system.prepareMonsterRoll("monster-damage", attackKey, undefined, undefined, undefined, defenderId, defenderTokenId, extraShieldDr)
@@ -394,7 +424,7 @@ Hooks.on("preCreateChatMessage", (message) => {
const rollType = message.rolls[0]?.options?.rollType
// Si c'est un message de défense et qu'on a des données en attente
if ((rollType === "weapon-defense" || rollType === "monster-defense") && game.lethalFantasy?.nextDefenseData) {
if ((rollType === "weapon-defense" || rollType === "monster-defense" || rollType === "save") && game.lethalFantasy?.nextDefenseData) {
// Ajouter les données dans les flags du message
message.updateSource({
[`flags.${SYSTEM.id}.attackData`]: game.lethalFantasy.nextDefenseData
@@ -414,7 +444,7 @@ Hooks.on("createChatMessage", async (message) => {
console.log("Defense hook checking message, rollType:", rollType)
// Vérifier si c'est un message de défense
if (rollType !== "weapon-defense" && rollType !== "monster-defense") return
if (rollType !== "weapon-defense" && rollType !== "monster-defense" && rollType !== "save") return
// Récupérer les données d'attaque depuis les flags
const attackData = message.flags?.[SYSTEM.id]?.attackData
@@ -483,7 +513,9 @@ Hooks.on("createChatMessage", async (message) => {
// Seulement si l'utilisateur actuel est le propriétaire du défenseur
let defenderHandledBonus = false
let shieldReaction = null
if (defender && defenseRoll < attackRoll && isPrimaryController(defender)) {
let shieldBlocked = false
const isSpellOrMiracle = attackRollType === "spell-attack" || attackRollType === "miracle-attack"
if (defender && defenseRoll < attackRoll && isPrimaryController(defender) && !isSpellOrMiracle) {
const shieldData = LethalFantasyUtils.getShieldReactionData(defender)
let canRerollDefense = LethalFantasyUtils.hasD30Reroll(defenseD30message)
let canShieldReact = !!shieldData
@@ -598,17 +630,30 @@ Hooks.on("createChatMessage", async (message) => {
if (choice === "shieldReact" && canShieldReact) {
const shieldBonus = await LethalFantasyUtils.rollBonusDie(shieldData.formula, defender)
defenseRoll += shieldBonus
const newDefenseTotal = defenseRoll + shieldBonus
defenseRoll = newDefenseTotal
canShieldReact = false
if (newDefenseTotal >= attackRoll) {
// Shield roll tied or exceeded the attack — shield blocked
shieldBlocked = true
shieldReaction = {
damageReduction: shieldData.damageReduction,
label: shieldData.label,
bonus: shieldBonus
}
canShieldReact = false
await createReactionMessage(
defender,
`<p><strong>${defenderName}</strong> rolls <strong>${shieldData.label}</strong> and adds <strong>${shieldBonus}</strong> to defense.${defenseRoll < attackRoll ? ` The hit still lands, but shield DR <strong>${shieldData.damageReduction}</strong> will reduce the damage.` : ""}</p>`
`<p><strong>${defenderName}</strong> rolls <strong>${shieldData.label}</strong> and adds <strong>${shieldBonus}</strong> to defense (${newDefenseTotal}${attackRoll}). <strong>Shield blocked the attack!</strong> Both armor DR and shield DR <strong>${shieldData.damageReduction}</strong> will apply to damage.</p>`
)
} else {
// Shield roll not enough — hit still lands, armor DR only
shieldReaction = null
await createReactionMessage(
defender,
`<p><strong>${defenderName}</strong> rolls <strong>${shieldData.label}</strong> and adds <strong>${shieldBonus}</strong> to defense (${newDefenseTotal} < ${attackRoll}). Shield did not block — normal hit, armor DR only.</p>`
)
}
}
}
}
@@ -704,8 +749,8 @@ Hooks.on("createChatMessage", async (message) => {
}
}
const shieldDamageReduction = shieldReaction && attackRollFinal > defenseRoll ? shieldReaction.damageReduction : 0
const outcome = shieldDamageReduction > 0 ? "shielded-hit" : (attackRollFinal > defenseRoll ? "hit" : "miss")
const shieldDamageReduction = shieldBlocked ? shieldReaction.damageReduction : 0
const outcome = shieldBlocked ? "shielded-hit" : (attackRollFinal > defenseRoll ? "hit" : "miss")
// Créer le message de comparaison - uniquement par le client qui a géré le dernier bonus
// Priorité: attaquant si il a géré le bonus, sinon défenseur si il a géré le bonus, sinon défenseur
@@ -788,7 +833,8 @@ Hooks.on("createChatMessage", async (message) => {
const attackerName = message.rolls[0]?.options?.actorName || "Unknown Attacker"
// Calculer les DR
const armorDR = defender.computeDamageReduction() || 0
const bypassArmor = message.rolls[0]?.options?.bypassArmor || false
const armorDR = bypassArmor ? 0 : (defender.computeDamageReduction() || 0)
const extraShieldDr = Number(message.rolls[0]?.options?.extraShieldDr) || 0
const totalDR = armorDR + extraShieldDr
const finalDamage = Math.max(0, damageTotal - totalDR)
@@ -823,7 +869,7 @@ Hooks.on("createChatMessage", async (message) => {
{
targetName: defender.name,
damage: finalDamage,
drText: totalDR > 0 ? `Armor DR: ${armorDR}${extraShieldDr > 0 ? ` + Shield DR: ${extraShieldDr}` : ""}` : "",
drText: bypassArmor ? "Armor DR bypassed (spell)" : (totalDR > 0 ? `Armor DR: ${armorDR}${extraShieldDr > 0 ? ` + Shield DR: ${extraShieldDr}` : ""}` : ""),
weaponName: weaponName,
attackerName: attackerName,
rawDamage: damageTotal

View File

@@ -39,6 +39,7 @@ export default class LethalFantasySpell extends foundry.abstract.TypeDataModel {
schema.attackRoll = new fields.StringField({ required: true, initial: "" })
schema.powerRoll = new fields.StringField({ required: true, initial: "" })
schema.damageDice = new fields.StringField({ required: false, initial: "" })
return schema
}

View File

@@ -235,6 +235,70 @@ export default class LethalFantasyUtils {
const isMonster = defender.type === "monster"
// Spell/miracle attacks use saving throws instead of weapon defense
const isSpellAttack = attackRollType === "spell-attack" || attackRollType === "miracle-attack"
if (isSpellAttack) {
const savesConfig = isMonster ? SYSTEM.MONSTER_SAVES : SYSTEM.SAVES
const combatSaves = ["will", "dodge", "toughness"]
const savesHTML = Object.values(savesConfig)
.filter(s => combatSaves.includes(s.id))
.map(s => `<option value="${s.id}">${game.i18n.localize(s.label)}</option>`)
.join("")
const content = `
<div class="defense-request-dialog">
<div class="attack-info">
<p><strong>${attackerName}</strong> targets <strong>${defenderName}</strong> with <strong>${weaponName}</strong>!</p>
<p>Attack roll: <strong>${attackRoll}</strong></p>
</div>
<div class="weapon-selection">
<label for="save-type">Choose saving throw:</label>
<select id="save-type" name="saveKey" style="width: 100%; margin-top: 8px;">
${savesHTML}
</select>
</div>
</div>
`
const result = await foundry.applications.api.DialogV2.wait({
window: { title: "Saving Throw vs Spell" },
classes: ["lethalfantasy"],
content,
buttons: [
{
label: "Roll Save",
icon: "fa-solid fa-person-running",
callback: (event, button) => button.form.elements.saveKey.value,
},
],
rejectClose: false
})
if (result) {
game.lethalFantasy = game.lethalFantasy || {}
game.lethalFantasy.nextDefenseData = {
attackerId,
attackRoll,
attackerName,
defenderName,
attackWeaponId,
attackRollType,
attackRollKey,
attackD30result,
attackD30message,
attackRerollContext,
defenderId: defender.id,
defenderTokenId
}
if (isMonster) {
defender.system.prepareMonsterRoll("save", result)
} else {
defender.prepareRoll("save", result)
}
}
return
}
// Pour les monstres, récupérer les attaques activées
if (isMonster) {
const enabledAttacks = Object.entries(defender.system.attacks).filter(([key, attack]) => attack.enabled)
@@ -440,11 +504,16 @@ export default class LethalFantasyUtils {
content,
buttons: [
{
action: "roll",
label: "Roll Bonus Die",
icon: "fa-solid fa-dice",
callback: (event, button, dialog) => button.form.elements.bonusDie.value
callback: (event, button) => {
const sel = button.form?.elements?.bonusDie ?? button.closest("form")?.elements?.bonusDie
return sel?.value ?? choices[0]
}
},
{
action: "cancel",
label: "Cancel",
icon: "fa-solid fa-xmark",
callback: () => null
@@ -684,6 +753,24 @@ export default class LethalFantasyUtils {
</button>
</div>
`
} else if (data.attackRollType === "spell-attack" || data.attackRollType === "miracle-attack") {
const attacker = game.actors.get(data.attackerId)
const spell = attacker?.items.get(data.attackWeaponId)
const damageDice = spell?.system?.damageDice
if (damageDice) {
damageButton = `
<div class="attack-result-damage">
<button class="roll-damage-btn"
data-attacker-id="${data.attackerId}"
data-defender-id="${data.defenderId}"
data-defender-token-id="${data.defenderTokenId || ""}"
data-damage-type="spell"
data-damage-formula="${damageDice}">
<i class="fa-solid fa-wand-magic-sparkles"></i> Spell Damage (${damageDice})
</button>
</div>
`
}
}
}
@@ -709,7 +796,7 @@ export default class LethalFantasyUtils {
</div>
<div class="combat-result-text">
${outcome === "shielded-hit"
? `<i class="fa-solid fa-shield"></i> <strong>${data.attackerName}</strong> still hits <strong>${data.defenderName}</strong>, but shield DR <strong>${data.shieldDamageReduction || 0}</strong> reduces the damage.`
? `<i class="fa-solid fa-shield"></i> <strong>${data.attackerName}</strong> hits <strong>${data.defenderName}</strong>, but the shield blocked — apply armor DR + shield DR <strong>${data.shieldDamageReduction || 0}</strong>.`
: isAttackWin
? `<i class="fa-solid fa-circle-check"></i> <strong>${data.attackerName}</strong> hits <strong>${data.defenderName}</strong>!`
: `<i class="fa-solid fa-shield-halved"></i> <strong>${data.defenderName}</strong> parries the attack!`

View File

@@ -1 +1 @@
MANIFEST-000555
MANIFEST-000559

View File

@@ -1,8 +1,8 @@
2026/04/12-10:10:03.851531 7f20ed3fd6c0 Recovering log #553
2026/04/12-10:10:03.861282 7f20ed3fd6c0 Delete type=3 #551
2026/04/12-10:10:03.861341 7f20ed3fd6c0 Delete type=0 #553
2026/04/12-11:08:03.389691 7f1e4ffff6c0 Level-0 table #558: started
2026/04/12-11:08:03.389742 7f1e4ffff6c0 Level-0 table #558: 0 bytes OK
2026/04/12-11:08:03.395757 7f1e4ffff6c0 Delete type=0 #556
2026/04/12-11:08:03.415391 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.415438 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end)
2026/04/14-20:54:14.172569 7fc1dafbf6c0 Recovering log #557
2026/04/14-20:54:14.184089 7fc1dafbf6c0 Delete type=3 #555
2026/04/14-20:54:14.184186 7fc1dafbf6c0 Delete type=0 #557
2026/04/14-21:31:09.240081 7fc1d8fbb6c0 Level-0 table #562: started
2026/04/14-21:31:09.240104 7fc1d8fbb6c0 Level-0 table #562: 0 bytes OK
2026/04/14-21:31:09.273756 7fc1d8fbb6c0 Delete type=0 #560
2026/04/14-21:31:09.274050 7fc1d8fbb6c0 Manual compaction at level-0 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end)
2026/04/14-21:31:09.274107 7fc1d8fbb6c0 Manual compaction at level-1 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end)

View File

@@ -1,8 +1,8 @@
2026/04/12-00:56:41.816652 7f20ecbfc6c0 Recovering log #549
2026/04/12-00:56:41.832396 7f20ecbfc6c0 Delete type=3 #547
2026/04/12-00:56:41.832446 7f20ecbfc6c0 Delete type=0 #549
2026/04/12-01:07:05.182307 7f1e4ffff6c0 Level-0 table #554: started
2026/04/12-01:07:05.182364 7f1e4ffff6c0 Level-0 table #554: 0 bytes OK
2026/04/12-01:07:05.188459 7f1e4ffff6c0 Delete type=0 #552
2026/04/12-01:07:05.207103 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end)
2026/04/12-01:07:05.207149 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end)
2026/04/12-10:10:03.851531 7f20ed3fd6c0 Recovering log #553
2026/04/12-10:10:03.861282 7f20ed3fd6c0 Delete type=3 #551
2026/04/12-10:10:03.861341 7f20ed3fd6c0 Delete type=0 #553
2026/04/12-11:08:03.389691 7f1e4ffff6c0 Level-0 table #558: started
2026/04/12-11:08:03.389742 7f1e4ffff6c0 Level-0 table #558: 0 bytes OK
2026/04/12-11:08:03.395757 7f1e4ffff6c0 Delete type=0 #556
2026/04/12-11:08:03.415391 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.415438 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end)

View File

@@ -1 +1 @@
MANIFEST-000552
MANIFEST-000556

View File

@@ -1,8 +1,8 @@
2026/04/12-10:10:03.864565 7f20ecbfc6c0 Recovering log #550
2026/04/12-10:10:03.875268 7f20ecbfc6c0 Delete type=3 #548
2026/04/12-10:10:03.875332 7f20ecbfc6c0 Delete type=0 #550
2026/04/12-11:08:03.395874 7f1e4ffff6c0 Level-0 table #555: started
2026/04/12-11:08:03.395896 7f1e4ffff6c0 Level-0 table #555: 0 bytes OK
2026/04/12-11:08:03.401955 7f1e4ffff6c0 Delete type=0 #553
2026/04/12-11:08:03.415404 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.415431 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end)
2026/04/14-20:54:14.189613 7fc1d9fbd6c0 Recovering log #554
2026/04/14-20:54:14.199454 7fc1d9fbd6c0 Delete type=3 #552
2026/04/14-20:54:14.199515 7fc1d9fbd6c0 Delete type=0 #554
2026/04/14-21:31:09.274309 7fc1d8fbb6c0 Level-0 table #559: started
2026/04/14-21:31:09.274344 7fc1d8fbb6c0 Level-0 table #559: 0 bytes OK
2026/04/14-21:31:09.316123 7fc1d8fbb6c0 Delete type=0 #557
2026/04/14-21:31:09.426109 7fc1d8fbb6c0 Manual compaction at level-0 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end)
2026/04/14-21:31:09.426138 7fc1d8fbb6c0 Manual compaction at level-1 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end)

View File

@@ -1,8 +1,8 @@
2026/04/12-00:56:41.835571 7f20ee3ff6c0 Recovering log #546
2026/04/12-00:56:41.850974 7f20ee3ff6c0 Delete type=3 #544
2026/04/12-00:56:41.851033 7f20ee3ff6c0 Delete type=0 #546
2026/04/12-01:07:05.194434 7f1e4ffff6c0 Level-0 table #551: started
2026/04/12-01:07:05.194455 7f1e4ffff6c0 Level-0 table #551: 0 bytes OK
2026/04/12-01:07:05.200694 7f1e4ffff6c0 Delete type=0 #549
2026/04/12-01:07:05.207125 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end)
2026/04/12-01:07:05.207165 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end)
2026/04/12-10:10:03.864565 7f20ecbfc6c0 Recovering log #550
2026/04/12-10:10:03.875268 7f20ecbfc6c0 Delete type=3 #548
2026/04/12-10:10:03.875332 7f20ecbfc6c0 Delete type=0 #550
2026/04/12-11:08:03.395874 7f1e4ffff6c0 Level-0 table #555: started
2026/04/12-11:08:03.395896 7f1e4ffff6c0 Level-0 table #555: 0 bytes OK
2026/04/12-11:08:03.401955 7f1e4ffff6c0 Delete type=0 #553
2026/04/12-11:08:03.415404 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.415431 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end)

View File

@@ -1 +1 @@
MANIFEST-000557
MANIFEST-000561

View File

@@ -1,8 +1,8 @@
2026/04/12-10:10:03.838632 7f20ee3ff6c0 Recovering log #555
2026/04/12-10:10:03.848360 7f20ee3ff6c0 Delete type=3 #553
2026/04/12-10:10:03.848406 7f20ee3ff6c0 Delete type=0 #555
2026/04/12-11:08:03.402029 7f1e4ffff6c0 Level-0 table #560: started
2026/04/12-11:08:03.402048 7f1e4ffff6c0 Level-0 table #560: 0 bytes OK
2026/04/12-11:08:03.408790 7f1e4ffff6c0 Delete type=0 #558
2026/04/12-11:08:03.415414 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.415454 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end)
2026/04/14-20:54:14.156150 7fc1d97bc6c0 Recovering log #559
2026/04/14-20:54:14.166976 7fc1d97bc6c0 Delete type=3 #557
2026/04/14-20:54:14.167099 7fc1d97bc6c0 Delete type=0 #559
2026/04/14-21:31:09.203242 7fc1d8fbb6c0 Level-0 table #564: started
2026/04/14-21:31:09.203283 7fc1d8fbb6c0 Level-0 table #564: 0 bytes OK
2026/04/14-21:31:09.239957 7fc1d8fbb6c0 Delete type=0 #562
2026/04/14-21:31:09.274032 7fc1d8fbb6c0 Manual compaction at level-0 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end)
2026/04/14-21:31:09.274121 7fc1d8fbb6c0 Manual compaction at level-1 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end)

View File

@@ -1,8 +1,8 @@
2026/04/12-00:56:41.797453 7f20ed3fd6c0 Recovering log #551
2026/04/12-00:56:41.813315 7f20ed3fd6c0 Delete type=3 #549
2026/04/12-00:56:41.813371 7f20ed3fd6c0 Delete type=0 #551
2026/04/12-01:07:05.200810 7f1e4ffff6c0 Level-0 table #556: started
2026/04/12-01:07:05.200837 7f1e4ffff6c0 Level-0 table #556: 0 bytes OK
2026/04/12-01:07:05.207008 7f1e4ffff6c0 Delete type=0 #554
2026/04/12-01:07:05.207134 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end)
2026/04/12-01:07:05.207157 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end)
2026/04/12-10:10:03.838632 7f20ee3ff6c0 Recovering log #555
2026/04/12-10:10:03.848360 7f20ee3ff6c0 Delete type=3 #553
2026/04/12-10:10:03.848406 7f20ee3ff6c0 Delete type=0 #555
2026/04/12-11:08:03.402029 7f1e4ffff6c0 Level-0 table #560: started
2026/04/12-11:08:03.402048 7f1e4ffff6c0 Level-0 table #560: 0 bytes OK
2026/04/12-11:08:03.408790 7f1e4ffff6c0 Delete type=0 #558
2026/04/12-11:08:03.415414 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.415454 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end)

View File

@@ -1 +1 @@
MANIFEST-000252
MANIFEST-000256

View File

@@ -1,8 +1,8 @@
2026/04/12-10:10:03.889781 7f20ed3fd6c0 Recovering log #250
2026/04/12-10:10:03.899549 7f20ed3fd6c0 Delete type=3 #248
2026/04/12-10:10:03.899595 7f20ed3fd6c0 Delete type=0 #250
2026/04/12-11:08:03.421462 7f1e4ffff6c0 Level-0 table #255: started
2026/04/12-11:08:03.421486 7f1e4ffff6c0 Level-0 table #255: 0 bytes OK
2026/04/12-11:08:03.427527 7f1e4ffff6c0 Delete type=0 #253
2026/04/12-11:08:03.443589 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.443624 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end)
2026/04/14-20:54:14.216969 7fc1d97bc6c0 Recovering log #254
2026/04/14-20:54:14.226830 7fc1d97bc6c0 Delete type=3 #252
2026/04/14-20:54:14.226876 7fc1d97bc6c0 Delete type=0 #254
2026/04/14-21:31:09.316377 7fc1d8fbb6c0 Level-0 table #259: started
2026/04/14-21:31:09.316424 7fc1d8fbb6c0 Level-0 table #259: 0 bytes OK
2026/04/14-21:31:09.353574 7fc1d8fbb6c0 Delete type=0 #257
2026/04/14-21:31:09.426120 7fc1d8fbb6c0 Manual compaction at level-0 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end)
2026/04/14-21:31:09.426154 7fc1d8fbb6c0 Manual compaction at level-1 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end)

View File

@@ -1,8 +1,8 @@
2026/04/12-00:56:41.870438 7f20ecbfc6c0 Recovering log #246
2026/04/12-00:56:41.885745 7f20ecbfc6c0 Delete type=3 #244
2026/04/12-00:56:41.885813 7f20ecbfc6c0 Delete type=0 #246
2026/04/12-01:07:05.229253 7f1e4ffff6c0 Level-0 table #251: started
2026/04/12-01:07:05.229286 7f1e4ffff6c0 Level-0 table #251: 0 bytes OK
2026/04/12-01:07:05.235261 7f1e4ffff6c0 Delete type=0 #249
2026/04/12-01:07:05.235386 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end)
2026/04/12-01:07:05.245080 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end)
2026/04/12-10:10:03.889781 7f20ed3fd6c0 Recovering log #250
2026/04/12-10:10:03.899549 7f20ed3fd6c0 Delete type=3 #248
2026/04/12-10:10:03.899595 7f20ed3fd6c0 Delete type=0 #250
2026/04/12-11:08:03.421462 7f1e4ffff6c0 Level-0 table #255: started
2026/04/12-11:08:03.421486 7f1e4ffff6c0 Level-0 table #255: 0 bytes OK
2026/04/12-11:08:03.427527 7f1e4ffff6c0 Delete type=0 #253
2026/04/12-11:08:03.443589 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.443624 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end)

View File

@@ -1 +1 @@
MANIFEST-000551
MANIFEST-000555

View File

@@ -1,8 +1,8 @@
2026/04/12-10:10:03.877445 7f20ee3ff6c0 Recovering log #549
2026/04/12-10:10:03.887615 7f20ee3ff6c0 Delete type=3 #547
2026/04/12-10:10:03.887678 7f20ee3ff6c0 Delete type=0 #549
2026/04/12-11:08:03.408913 7f1e4ffff6c0 Level-0 table #554: started
2026/04/12-11:08:03.408942 7f1e4ffff6c0 Level-0 table #554: 0 bytes OK
2026/04/12-11:08:03.415303 7f1e4ffff6c0 Delete type=0 #552
2026/04/12-11:08:03.415422 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.415446 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end)
2026/04/14-20:54:14.202677 7fc1da7be6c0 Recovering log #553
2026/04/14-20:54:14.213358 7fc1da7be6c0 Delete type=3 #551
2026/04/14-20:54:14.213482 7fc1da7be6c0 Delete type=0 #553
2026/04/14-21:31:09.160202 7fc1d8fbb6c0 Level-0 table #558: started
2026/04/14-21:31:09.160283 7fc1d8fbb6c0 Level-0 table #558: 0 bytes OK
2026/04/14-21:31:09.203034 7fc1d8fbb6c0 Delete type=0 #556
2026/04/14-21:31:09.274014 7fc1d8fbb6c0 Manual compaction at level-0 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end)
2026/04/14-21:31:09.274091 7fc1d8fbb6c0 Manual compaction at level-1 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end)

View File

@@ -1,8 +1,8 @@
2026/04/12-00:56:41.853085 7f20ed3fd6c0 Recovering log #545
2026/04/12-00:56:41.867862 7f20ed3fd6c0 Delete type=3 #543
2026/04/12-00:56:41.867925 7f20ed3fd6c0 Delete type=0 #545
2026/04/12-01:07:05.188580 7f1e4ffff6c0 Level-0 table #550: started
2026/04/12-01:07:05.188606 7f1e4ffff6c0 Level-0 table #550: 0 bytes OK
2026/04/12-01:07:05.194367 7f1e4ffff6c0 Delete type=0 #548
2026/04/12-01:07:05.207115 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end)
2026/04/12-01:07:05.207142 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end)
2026/04/12-10:10:03.877445 7f20ee3ff6c0 Recovering log #549
2026/04/12-10:10:03.887615 7f20ee3ff6c0 Delete type=3 #547
2026/04/12-10:10:03.887678 7f20ee3ff6c0 Delete type=0 #549
2026/04/12-11:08:03.408913 7f1e4ffff6c0 Level-0 table #554: started
2026/04/12-11:08:03.408942 7f1e4ffff6c0 Level-0 table #554: 0 bytes OK
2026/04/12-11:08:03.415303 7f1e4ffff6c0 Delete type=0 #552
2026/04/12-11:08:03.415422 7f1e4ffff6c0 Manual compaction at level-0 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end)
2026/04/12-11:08:03.415446 7f1e4ffff6c0 Manual compaction at level-1 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end)

View File

@@ -30,6 +30,7 @@
{{formField systemFields.savingThrow value=system.savingThrow}}
{{formField systemFields.extraAetherPoints value=system.extraAetherPoints}}
{{formField systemFields.criticalType value=system.criticalType}}
{{formField systemFields.damageDice value=system.damageDice}}
<fieldset>