From 38eb1a8d3de6915065c058c54ff5e304d7e77da4 Mon Sep 17 00:00:00 2001 From: LeRatierBretonnier Date: Sat, 23 May 2026 19:10:10 +0200 Subject: [PATCH] Add ranged actions for monsters --- CLAUDE.md | 62 +++++++ lang/en.json | 2 + module/config/system.mjs | 8 + module/documents/roll.mjs | 157 ++++++++++++++++++ module/models/character.mjs | 17 ++ module/models/monster.mjs | 31 ++++ .../lf-equipment/{000613.log => 000617.log} | 0 packs-system/lf-equipment/CURRENT | 2 +- packs-system/lf-equipment/LOG | 16 +- packs-system/lf-equipment/LOG.old | 16 +- .../{MANIFEST-000611 => MANIFEST-000615} | Bin 178 -> 178 bytes .../lf-gifts/{000610.log => 000614.log} | 0 packs-system/lf-gifts/CURRENT | 2 +- packs-system/lf-gifts/LOG | 16 +- packs-system/lf-gifts/LOG.old | 16 +- .../{MANIFEST-000608 => MANIFEST-000612} | Bin 247 -> 247 bytes .../lf-skills/{000615.log => 000619.log} | 0 packs-system/lf-skills/CURRENT | 2 +- packs-system/lf-skills/LOG | 16 +- packs-system/lf-skills/LOG.old | 16 +- .../{MANIFEST-000613 => MANIFEST-000617} | Bin 178 -> 178 bytes .../{000310.log => 000314.log} | 0 packs-system/lf-spells-miracles/CURRENT | 2 +- packs-system/lf-spells-miracles/LOG | 16 +- packs-system/lf-spells-miracles/LOG.old | 16 +- .../{MANIFEST-000308 => MANIFEST-000312} | Bin 177 -> 177 bytes .../{000609.log => 000613.log} | 0 packs-system/lf-vulnerabilities/CURRENT | 2 +- packs-system/lf-vulnerabilities/LOG | 16 +- packs-system/lf-vulnerabilities/LOG.old | 16 +- .../{MANIFEST-000607 => MANIFEST-000611} | Bin 176 -> 176 bytes templates/range-attack-dialog.hbs | 45 +++++ 32 files changed, 407 insertions(+), 85 deletions(-) create mode 100644 CLAUDE.md rename packs-system/lf-equipment/{000613.log => 000617.log} (100%) rename packs-system/lf-equipment/{MANIFEST-000611 => MANIFEST-000615} (71%) rename packs-system/lf-gifts/{000610.log => 000614.log} (100%) rename packs-system/lf-gifts/{MANIFEST-000608 => MANIFEST-000612} (77%) rename packs-system/lf-skills/{000615.log => 000619.log} (100%) rename packs-system/lf-skills/{MANIFEST-000613 => MANIFEST-000617} (71%) rename packs-system/lf-spells-miracles/{000310.log => 000314.log} (100%) rename packs-system/lf-spells-miracles/{MANIFEST-000308 => MANIFEST-000312} (72%) rename packs-system/lf-vulnerabilities/{000609.log => 000613.log} (100%) rename packs-system/lf-vulnerabilities/{MANIFEST-000607 => MANIFEST-000611} (72%) create mode 100644 templates/range-attack-dialog.hbs diff --git a/CLAUDE.md b/CLAUDE.md new file mode 100644 index 0000000..869e187 --- /dev/null +++ b/CLAUDE.md @@ -0,0 +1,62 @@ +# CLAUDE.md + +This file provides guidance to Claude Code (claude.ai/code) when working with this repository. + +## Overview + +FoundryVTT v13+ game system for the **Lethal Fantasy RPG**. Entry point: `lethal-fantasy.mjs`. + +## Commands + +```bash +# Compile LESS styles (styles/ -> css/) +npx gulp css # one-shot +npx gulp # compile + watch + +# Lint +npx eslint . + +# Compendium pack management (LevelDB <-> YAML source) +npm run pushLDBtoYML # export packs-system/ LevelDB -> source YAML +npm run pullYMLtoLDB # import source YAML -> packs-system/ LevelDB +``` + +No test suite exists. + +## Architecture + +Four layers in `module/`, all wired in `lethal-fantasy.mjs` via the `init` hook: + +| Layer | Path | Purpose | +|---|---|---| +| Config | `module/config/` | Game constants. `SYSTEM` is `globalThis.SYSTEM` — always use `SYSTEM.*` for enumerations. | +| Models | `module/models/` | `TypeDataModel` subclasses — data schemas per document type. | +| Documents | `module/documents/` | Actor/Item/Roll/ChatMessage subclasses — game logic, roll processing, hooks. | +| Applications | `module/applications/` | `ApplicationV2` sheets + custom combat tracker. | + +**Actor types**: `character`, `monster` +**Item types**: `skill`, `gift`, `vulnerability`, `weapon`, `armor`, `shield`, `spell`, `miracle`, `equipment` + +Each layer has an `_module.mjs` barrel file that re-exports all classes from that layer. + +Templates (`.hbs`) live in `templates/`. Styles are authored in LESS under `styles/` and compiled to `css/`. + +### Key Patterns + +- **Sheets**: Extend `HandlebarsApplicationMixin(foundry.applications.sheets.ActorSheetV2)` — imported from `foundry.applications.api`. **Not** the legacy `ActorSheet`. Child sheets (e.g. `character-sheet.mjs`) extend `base-actor-sheet.mjs` and override `static PARTS` and `DEFAULT_OPTIONS.actions`. Template paths are prefixed `systems/fvtt-lethal-fantasy/templates/`. Actor sheets have a play/edit toggle via `_sheetMode` and `SHEET_MODES`. +- **Models**: `static defineSchema()` using `foundry.data.fields.*`. Field definitions derived from SYSTEM config objects. +- **Rolls**: `LethalFantasyRoll` extends `Roll` with rich metadata via `this.options`. `D30Roll` is a separate class for D30 result tables (initialized in the `ready` hook). +- **Socket**: Events use `game.socket.on(\`system.${SYSTEM.id}\`, ...)`. Multi-player attack-defense uses a global `pendingDefenses` Map. +- **i18n**: All user-visible strings are keys in `lang/en.json` as `LETHALFANTASY.Category.Key`. Always use `game.i18n.localize(key)`. + +### Compendium Packs + +Five LevelDB packs in `packs-system/`: skills, equipment, gifts, vulnerabilities, spells-miracles. Use the `tools/` scripts to export/import editable YAML. + +## Code Style + +- No semicolons, double quotes, 2-space indent +- JSDoc `/** */` required on all functions/classes +- Max line length 180 (strings/templates exempt) +- Arrow functions: omit parens for single param +- ESLint + Prettier config in `eslint.config.mjs` diff --git a/lang/en.json b/lang/en.json index 7aa69bc..9571b7c 100644 --- a/lang/en.json +++ b/lang/en.json @@ -471,6 +471,8 @@ "range": "Range", "rangeDefenseDialog": "Ranged defense dialog", "rangeDefenseRoll": "Ranged defense roll", + "rangeAttackDialog": "Ranged attack dialog", + "rangeAttackRoll": "Ranged attack roll", "rangedAttackDefense": "Ranged attack defense", "resource": "Resource", "resources": "Resources", diff --git a/module/config/system.mjs b/module/config/system.mjs index e590b2d..de5a6ea 100644 --- a/module/config/system.mjs +++ b/module/config/system.mjs @@ -90,6 +90,13 @@ export const RANGE_CHOICES = { "beyondskill": { label: "Beyond Skill (Blue +11)", value: "beyondskill" } } +export const ATTACKER_MOVEMENT_CHOICES = { + "none": { label: "None / Stationary (D20E Favor)", favor: true, value: "2D20kh" }, + "walk": { label: "Walk (D20E)", value: "D20" }, + "incombat": { label: "In Combat (D20E)", value: "D20" }, + "run": { label: "Jog/Run/Sprint (D20E Disfavor)", disfavor: true, value: "2D20kl" } +} + export const ATTACKER_AIM_CHOICES = { "simple": { label: "Simple (+0)", value: "0" }, "careful": { label: "Careful (Red +5)", value: "+4" }, @@ -321,6 +328,7 @@ export const SYSTEM = { RANGE_CHOICES, FAVOR_CHOICES, ATTACKER_AIM_CHOICES, + ATTACKER_MOVEMENT_CHOICES, MORTAL_CHOICES, SPELL_CRITICAL, MIRACLE_TYPES, diff --git a/module/documents/roll.mjs b/module/documents/roll.mjs index f08ae29..9c37bbd 100644 --- a/module/documents/roll.mjs +++ b/module/documents/roll.mjs @@ -1133,6 +1133,8 @@ export default class LethalFantasyRoll extends Roll { options = { ...options, ...rollContext } options.rollName = "Ranged Defense" options.rollType = "weapon-defense" + options.type = options.rollType // Required: this.type reads options.type + options.rollMode = rollContext.visibility // Required: callers pass roll.options.rollMode to toMessage const rollBase = new this(rollContext.movement, options.data, rollData) const rollModifier = new Roll(modifierFormula, options.data, rollData) @@ -1200,6 +1202,161 @@ export default class LethalFantasyRoll extends Roll { return rollBase } + /** + * Prompts the GM for ranged attack context (movement, range, target size, aim) when a monster + * attacks with a ranged weapon, then evaluates an exploding D20 attack roll with the resulting modifiers. + * + * @param {Object} options Options for the roll. + * @param {string} options.actorId The attacker actor ID. + * @param {string} options.actorName The attacker actor name. + * @param {Object} options.rollTarget The rollTarget containing attackModifier and related data. + * @returns {Promise} The resulting roll, or null if cancelled. + */ + static async promptRangedAttack(options = {}) { + const rollModes = foundry.utils.duplicate(CONFIG.ChatMessage.modes) + const fieldRollMode = new foundry.data.fields.StringField({ + choices: rollModes, + blank: false, + default: "public", + }) + + let dialogContext = { + attackerMovementChoices: SYSTEM.ATTACKER_MOVEMENT_CHOICES, + rangeChoices: SYSTEM.RANGE_CHOICES, + sizeChoices: SYSTEM.SIZE_CHOICES, + attackerAimChoices: SYSTEM.ATTACKER_AIM_CHOICES, + movement: "none", + range: "short", + size: "+5", + attackerAim: "simple", + fieldRollMode, + rollModes + } + + const content = await foundry.applications.handlebars.renderTemplate("systems/fvtt-lethal-fantasy/templates/range-attack-dialog.hbs", dialogContext) + + const label = game.i18n.localize("LETHALFANTASY.Label.rangeAttackRoll") + const rollContext = await foundry.applications.api.DialogV2.wait({ + window: { title: "Ranged Attack" }, + classes: ["lethalfantasy"], + content, + buttons: [ + { + label, + callback: (event, button) => { + const output = Array.from(button.form.elements).reduce((obj, input) => { + if (input.name) obj[input.name] = input.value + return obj + }, {}) + return output + }, + }, + ], + rejectClose: false + }) + + if (rollContext === null) return null + + // Handle pointblank: attacker at point blank gets favor (standing still easier to aim) + if (rollContext.range === "pointblank") { + rollContext.movement = rollContext.movement.replace("kh", "") + rollContext.movement = rollContext.movement.replace("kl", "") + rollContext.movement += "kh" // Favor for attacker at point blank + rollContext.range = "0" + } + // Handle beyondskill: extreme range gives disfavor to attacker + if (rollContext.range === "beyondskill") { + rollContext.movement = rollContext.movement.replace("kh", "") + rollContext.movement = rollContext.movement.replace("kl", "") + rollContext.movement += "kl" // Disfavor for attacker beyond skill range + rollContext.range = "+11" + } + + // Compute contextual penalty: range + target_size, reduced by aim bonus and attack modifier + const attackModifier = options.rollTarget?.attackModifier ?? 0 + const contextualPenalty = Number(rollContext.range) + Number(rollContext.size) + const aimBonus = Number(rollContext.attackerAim || 0) + const fullModifier = contextualPenalty - aimBonus - attackModifier + + let modifierFormula + if (fullModifier === 0) { + modifierFormula = "0" + } else { + const modAbs = Math.abs(fullModifier) + modifierFormula = `D${modAbs + 1} -1` + } + + const rollData = { ...rollContext } + options = { ...options, ...rollContext } + options.rollName = "Ranged Attack" + options.rollType = options.rollType || "monster-attack" + options.type = options.rollType // Required: this.type reads options.type, used to build weaponDamageOptions in toHTML + options.rollMode = rollContext.visibility // Required: callers pass roll.options.rollMode to toMessage + options.isRangedAttack = true + + const rollBase = new this(rollContext.movement, options.data, rollData) + const rollModifier = new Roll(modifierFormula, options.data, rollData) + rollModifier.evaluate() + await rollBase.evaluate() + const rollD30 = await new Roll("1D30").evaluate() + options.D30result = rollD30.total + options.D30message = D30Roll.getResult(rollD30.total, options.rollType, undefined, { isRanged: true }) + + // Determine favor from dice formula + let badResult = 0 + if (rollContext.movement.includes("kh")) { + rollData.favor = "favor" + badResult = Math.min(rollBase.terms[0].results[0].result, rollBase.terms[0].results[1]?.result || 20) + } + if (rollContext.movement.includes("kl")) { + rollData.favor = "disfavor" + badResult = Math.max(rollBase.terms[0].results[0].result, rollBase.terms[0].results[1]?.result || 1) + } + + const dice = rollContext.movement + const maxValue = 20 + let rollTotal = -1 + let diceResults = [] + + let diceResult = rollBase.dice[0].results[0].result + diceResults.push({ dice: `${dice.toUpperCase()}`, value: diceResult }) + let diceSum = diceResult + // Exploding dice + while (diceResult === maxValue) { + const r = await new Roll(dice).evaluate() + diceResult = r.dice[0].results[0].result + diceResults.push({ dice: `${dice.toUpperCase()}-1`, value: diceResult - 1 }) + diceSum += (diceResult - 1) + } + + if (fullModifier !== 0) { + diceResults.push({ dice: `${rollModifier.formula.toUpperCase()}`, value: rollModifier.total }) + if (fullModifier > 0) { + // Net penalty: subtract from roll + rollTotal = Math.max(diceSum - rollModifier.total, 0) + } else { + // Net bonus: add to roll + rollTotal = diceSum + rollModifier.total + } + } else { + rollTotal = diceSum + } + + rollBase.options = { ...rollBase.options, ...options } + rollBase.options.resultType = undefined + rollBase.options.rollTotal = rollTotal + rollBase.options.diceResults = diceResults + rollBase.options.rollTarget = options.rollTarget + rollBase.options.titleFormula = `1D20E + ${modifierFormula}` + rollBase.options.D30result = options.D30result + rollBase.options.D30message = options.D30message + rollBase.options.rollName = "Ranged Attack" + rollBase.options.badResult = badResult + rollBase.options.rollData = foundry.utils.duplicate(rollData) + + return rollBase + } + /** * Creates a title based on the given type. * diff --git a/module/models/character.mjs b/module/models/character.mjs index 3a53e42..fd0b356 100644 --- a/module/models/character.mjs +++ b/module/models/character.mjs @@ -283,6 +283,23 @@ export default class LethalFantasyCharacter extends foundry.abstract.TypeDataMod */ async roll(rollType, rollTarget, defenderId, defenderTokenId, extraShieldDr = 0) { const hasTarget = false + + // Ranged weapon attacks from PCs use the ranged attack dialog (range, movement, aim modifiers) + if (rollType === "weapon-attack" && rollTarget?.isRangedAttack === true) { + let roll = await LethalFantasyRoll.promptRangedAttack({ + rollType: "weapon-attack", + actorId: this.parent.id, + actorName: this.parent.name, + actorImage: this.parent.img, + rollTarget, + defenderId, + defenderTokenId, + }) + if (!roll) return null + await roll.toMessage({}, { messageMode: roll.options.rollMode }) + return + } + let roll = await LethalFantasyRoll.prompt({ rollType, rollTarget, diff --git a/module/models/monster.mjs b/module/models/monster.mjs index 1a1dcd9..035423c 100644 --- a/module/models/monster.mjs +++ b/module/models/monster.mjs @@ -166,6 +166,37 @@ export default class LethalFantasyMonster extends foundry.abstract.TypeDataModel */ async roll(rollType, rollTarget, defenderId = undefined, defenderTokenId = undefined, extraShieldDr = 0) { const hasTarget = false + + // Ranged monster attacks use a specialized dialog with range/movement/size/aim modifiers + if (rollType === "monster-attack" && rollTarget?.attackMode === "ranged") { + let roll = await LethalFantasyRoll.promptRangedAttack({ + actorId: this.parent.id, + actorName: this.parent.name, + actorImage: this.parent.img, + rollTarget, + defenderId, + defenderTokenId, + }) + if (!roll) return null + await roll.toMessage({}, { messageMode: roll.options.rollMode }) + return + } + + // Ranged monster defense uses the same ranged defense dialog as PC characters + if (rollType === "monster-defense" && rollTarget?.isRangedDefense === true) { + let roll = await LethalFantasyRoll.promptRangedDefense({ + actorId: this.parent.id, + actorName: this.parent.name, + actorImage: this.parent.img, + rollTarget, + defenderId, + defenderTokenId, + }) + if (!roll) return null + await roll.toMessage({}, { messageMode: roll.options.rollMode }) + return + } + let roll = await LethalFantasyRoll.prompt({ rollType, rollTarget, diff --git a/packs-system/lf-equipment/000613.log b/packs-system/lf-equipment/000617.log similarity index 100% rename from packs-system/lf-equipment/000613.log rename to packs-system/lf-equipment/000617.log diff --git a/packs-system/lf-equipment/CURRENT b/packs-system/lf-equipment/CURRENT index d0cc76d..40fde2e 100644 --- a/packs-system/lf-equipment/CURRENT +++ b/packs-system/lf-equipment/CURRENT @@ -1 +1 @@ -MANIFEST-000611 +MANIFEST-000615 diff --git a/packs-system/lf-equipment/LOG b/packs-system/lf-equipment/LOG index 69901ba..7fa5530 100644 --- a/packs-system/lf-equipment/LOG +++ b/packs-system/lf-equipment/LOG @@ -1,8 +1,8 @@ -2026/05/20-23:09:33.153266 7f179e7ed6c0 Recovering log #609 -2026/05/20-23:09:33.205459 7f179e7ed6c0 Delete type=3 #607 -2026/05/20-23:09:33.205507 7f179e7ed6c0 Delete type=0 #609 -2026/05/20-23:16:53.517842 7f179d7eb6c0 Level-0 table #614: started -2026/05/20-23:16:53.517893 7f179d7eb6c0 Level-0 table #614: 0 bytes OK -2026/05/20-23:16:53.523991 7f179d7eb6c0 Delete type=0 #612 -2026/05/20-23:16:53.536989 7f179d7eb6c0 Manual compaction at level-0 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end) -2026/05/20-23:16:53.537026 7f179d7eb6c0 Manual compaction at level-1 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end) +2026/05/23-18:45:10.214922 7fa233fff6c0 Recovering log #613 +2026/05/23-18:45:10.223944 7fa233fff6c0 Delete type=3 #611 +2026/05/23-18:45:10.223997 7fa233fff6c0 Delete type=0 #613 +2026/05/23-19:09:56.190877 7fa2327fc6c0 Level-0 table #618: started +2026/05/23-19:09:56.190922 7fa2327fc6c0 Level-0 table #618: 0 bytes OK +2026/05/23-19:09:56.224359 7fa2327fc6c0 Delete type=0 #616 +2026/05/23-19:09:56.339160 7fa2327fc6c0 Manual compaction at level-0 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end) +2026/05/23-19:09:56.339194 7fa2327fc6c0 Manual compaction at level-1 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-equipment/LOG.old b/packs-system/lf-equipment/LOG.old index 984adfb..69901ba 100644 --- a/packs-system/lf-equipment/LOG.old +++ b/packs-system/lf-equipment/LOG.old @@ -1,8 +1,8 @@ -2026/05/20-00:10:37.079181 7f0b1e7ee6c0 Recovering log #605 -2026/05/20-00:10:37.089613 7f0b1e7ee6c0 Delete type=3 #603 -2026/05/20-00:10:37.089671 7f0b1e7ee6c0 Delete type=0 #605 -2026/05/20-10:53:40.566713 7f0b1cfeb6c0 Level-0 table #610: started -2026/05/20-10:53:40.566740 7f0b1cfeb6c0 Level-0 table #610: 0 bytes OK -2026/05/20-10:53:40.572741 7f0b1cfeb6c0 Delete type=0 #608 -2026/05/20-10:53:40.587356 7f0b1cfeb6c0 Manual compaction at level-0 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end) -2026/05/20-10:53:40.587409 7f0b1cfeb6c0 Manual compaction at level-1 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end) +2026/05/20-23:09:33.153266 7f179e7ed6c0 Recovering log #609 +2026/05/20-23:09:33.205459 7f179e7ed6c0 Delete type=3 #607 +2026/05/20-23:09:33.205507 7f179e7ed6c0 Delete type=0 #609 +2026/05/20-23:16:53.517842 7f179d7eb6c0 Level-0 table #614: started +2026/05/20-23:16:53.517893 7f179d7eb6c0 Level-0 table #614: 0 bytes OK +2026/05/20-23:16:53.523991 7f179d7eb6c0 Delete type=0 #612 +2026/05/20-23:16:53.536989 7f179d7eb6c0 Manual compaction at level-0 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end) +2026/05/20-23:16:53.537026 7f179d7eb6c0 Manual compaction at level-1 from '!folders!ATr9wZhg5uTVTksM' @ 72057594037927935 : 1 .. '!items!zw9RQocTdz3HRjZK' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-equipment/MANIFEST-000611 b/packs-system/lf-equipment/MANIFEST-000615 similarity index 71% rename from packs-system/lf-equipment/MANIFEST-000611 rename to packs-system/lf-equipment/MANIFEST-000615 index 9e544a4f1afc770ec028c335e30d1e9d8743ae10..00d583da96946d14cc540eec5189942c976d6241 100644 GIT binary patch delta 43 tcmdnQxQTH>p9=S~H*wqyj7%?BI2o8pUOv-J-yrvj7(2hI2o9qvarnKJYnwY3KDq=5_t|10RSb-3%>vW diff --git a/packs-system/lf-gifts/000610.log b/packs-system/lf-gifts/000614.log similarity index 100% rename from packs-system/lf-gifts/000610.log rename to packs-system/lf-gifts/000614.log diff --git a/packs-system/lf-gifts/CURRENT b/packs-system/lf-gifts/CURRENT index 6cb8258..c116b38 100644 --- a/packs-system/lf-gifts/CURRENT +++ b/packs-system/lf-gifts/CURRENT @@ -1 +1 @@ -MANIFEST-000608 +MANIFEST-000612 diff --git a/packs-system/lf-gifts/LOG b/packs-system/lf-gifts/LOG index 64aa321..889e4a9 100644 --- a/packs-system/lf-gifts/LOG +++ b/packs-system/lf-gifts/LOG @@ -1,8 +1,8 @@ -2026/05/20-23:09:33.234730 7f179e7ed6c0 Recovering log #606 -2026/05/20-23:09:33.288868 7f179e7ed6c0 Delete type=3 #604 -2026/05/20-23:09:33.288920 7f179e7ed6c0 Delete type=0 #606 -2026/05/20-23:16:53.524172 7f179d7eb6c0 Level-0 table #611: started -2026/05/20-23:16:53.524219 7f179d7eb6c0 Level-0 table #611: 0 bytes OK -2026/05/20-23:16:53.530312 7f179d7eb6c0 Delete type=0 #609 -2026/05/20-23:16:53.537003 7f179d7eb6c0 Manual compaction at level-0 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end) -2026/05/20-23:16:53.537059 7f179d7eb6c0 Manual compaction at level-1 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end) +2026/05/23-18:45:10.228083 7fa232ffd6c0 Recovering log #610 +2026/05/23-18:45:10.236935 7fa232ffd6c0 Delete type=3 #608 +2026/05/23-18:45:10.236958 7fa232ffd6c0 Delete type=0 #610 +2026/05/23-19:09:56.261648 7fa2327fc6c0 Level-0 table #615: started +2026/05/23-19:09:56.261672 7fa2327fc6c0 Level-0 table #615: 0 bytes OK +2026/05/23-19:09:56.304021 7fa2327fc6c0 Delete type=0 #613 +2026/05/23-19:09:56.339181 7fa2327fc6c0 Manual compaction at level-0 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end) +2026/05/23-19:09:56.339208 7fa2327fc6c0 Manual compaction at level-1 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-gifts/LOG.old b/packs-system/lf-gifts/LOG.old index aba4aca..64aa321 100644 --- a/packs-system/lf-gifts/LOG.old +++ b/packs-system/lf-gifts/LOG.old @@ -1,8 +1,8 @@ -2026/05/20-00:10:37.107483 7f0b1efef6c0 Recovering log #602 -2026/05/20-00:10:37.117907 7f0b1efef6c0 Delete type=3 #600 -2026/05/20-00:10:37.117968 7f0b1efef6c0 Delete type=0 #602 -2026/05/20-10:53:40.572909 7f0b1cfeb6c0 Level-0 table #607: started -2026/05/20-10:53:40.572948 7f0b1cfeb6c0 Level-0 table #607: 0 bytes OK -2026/05/20-10:53:40.579001 7f0b1cfeb6c0 Delete type=0 #605 -2026/05/20-10:53:40.587370 7f0b1cfeb6c0 Manual compaction at level-0 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end) -2026/05/20-10:53:40.587430 7f0b1cfeb6c0 Manual compaction at level-1 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end) +2026/05/20-23:09:33.234730 7f179e7ed6c0 Recovering log #606 +2026/05/20-23:09:33.288868 7f179e7ed6c0 Delete type=3 #604 +2026/05/20-23:09:33.288920 7f179e7ed6c0 Delete type=0 #606 +2026/05/20-23:16:53.524172 7f179d7eb6c0 Level-0 table #611: started +2026/05/20-23:16:53.524219 7f179d7eb6c0 Level-0 table #611: 0 bytes OK +2026/05/20-23:16:53.530312 7f179d7eb6c0 Delete type=0 #609 +2026/05/20-23:16:53.537003 7f179d7eb6c0 Manual compaction at level-0 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end) +2026/05/20-23:16:53.537059 7f179d7eb6c0 Manual compaction at level-1 from '!folders!yPWGvxHJbDNHVSnY' @ 72057594037927935 : 1 .. '!items!x5gLtqlW4sdDmHTd' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-gifts/MANIFEST-000608 b/packs-system/lf-gifts/MANIFEST-000612 similarity index 77% rename from packs-system/lf-gifts/MANIFEST-000608 rename to packs-system/lf-gifts/MANIFEST-000612 index 4ccff253be1b0a2eb9b624b76324e11e0474ad35..10570ab4fdcf5df99c258b55a5fb8bb5c305136e 100644 GIT binary patch delta 43 tcmey)_?>aWWtD$nw`OxQFfu)5;bdTb#=^3j@o|Do4M^k}NaO`b1OQx#4S4_n delta 43 tcmey)_?>aWWflLcy9K!!7?~cja56AIVqw|Mn3Edu8zk}wB=Q6#0sujR4A}qx diff --git a/packs-system/lf-skills/000615.log b/packs-system/lf-skills/000619.log similarity index 100% rename from packs-system/lf-skills/000615.log rename to packs-system/lf-skills/000619.log diff --git a/packs-system/lf-skills/CURRENT b/packs-system/lf-skills/CURRENT index 25a6463..7b1d768 100644 --- a/packs-system/lf-skills/CURRENT +++ b/packs-system/lf-skills/CURRENT @@ -1 +1 @@ -MANIFEST-000613 +MANIFEST-000617 diff --git a/packs-system/lf-skills/LOG b/packs-system/lf-skills/LOG index 1ef041a..82d579e 100644 --- a/packs-system/lf-skills/LOG +++ b/packs-system/lf-skills/LOG @@ -1,8 +1,8 @@ -2026/05/20-23:09:33.082007 7f179dfec6c0 Recovering log #611 -2026/05/20-23:09:33.140910 7f179dfec6c0 Delete type=3 #609 -2026/05/20-23:09:33.140962 7f179dfec6c0 Delete type=0 #611 -2026/05/20-23:16:53.510592 7f179d7eb6c0 Level-0 table #616: started -2026/05/20-23:16:53.510664 7f179d7eb6c0 Level-0 table #616: 0 bytes OK -2026/05/20-23:16:53.517661 7f179d7eb6c0 Delete type=0 #614 -2026/05/20-23:16:53.536973 7f179d7eb6c0 Manual compaction at level-0 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end) -2026/05/20-23:16:53.537037 7f179d7eb6c0 Manual compaction at level-1 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end) +2026/05/23-18:45:10.200781 7fa2337fe6c0 Recovering log #615 +2026/05/23-18:45:10.211345 7fa2337fe6c0 Delete type=3 #613 +2026/05/23-18:45:10.211416 7fa2337fe6c0 Delete type=0 #615 +2026/05/23-19:09:56.304132 7fa2327fc6c0 Level-0 table #620: started +2026/05/23-19:09:56.304162 7fa2327fc6c0 Level-0 table #620: 0 bytes OK +2026/05/23-19:09:56.339034 7fa2327fc6c0 Delete type=0 #618 +2026/05/23-19:09:56.339189 7fa2327fc6c0 Manual compaction at level-0 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end) +2026/05/23-19:09:56.339203 7fa2327fc6c0 Manual compaction at level-1 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-skills/LOG.old b/packs-system/lf-skills/LOG.old index 9d3612c..1ef041a 100644 --- a/packs-system/lf-skills/LOG.old +++ b/packs-system/lf-skills/LOG.old @@ -1,8 +1,8 @@ -2026/05/20-00:10:37.061485 7f0b1efef6c0 Recovering log #607 -2026/05/20-00:10:37.071572 7f0b1efef6c0 Delete type=3 #605 -2026/05/20-00:10:37.071684 7f0b1efef6c0 Delete type=0 #607 -2026/05/20-10:53:40.579189 7f0b1cfeb6c0 Level-0 table #612: started -2026/05/20-10:53:40.579229 7f0b1cfeb6c0 Level-0 table #612: 0 bytes OK -2026/05/20-10:53:40.587176 7f0b1cfeb6c0 Delete type=0 #610 -2026/05/20-10:53:40.587386 7f0b1cfeb6c0 Manual compaction at level-0 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end) -2026/05/20-10:53:40.587419 7f0b1cfeb6c0 Manual compaction at level-1 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end) +2026/05/20-23:09:33.082007 7f179dfec6c0 Recovering log #611 +2026/05/20-23:09:33.140910 7f179dfec6c0 Delete type=3 #609 +2026/05/20-23:09:33.140962 7f179dfec6c0 Delete type=0 #611 +2026/05/20-23:16:53.510592 7f179d7eb6c0 Level-0 table #616: started +2026/05/20-23:16:53.510664 7f179d7eb6c0 Level-0 table #616: 0 bytes OK +2026/05/20-23:16:53.517661 7f179d7eb6c0 Delete type=0 #614 +2026/05/20-23:16:53.536973 7f179d7eb6c0 Manual compaction at level-0 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end) +2026/05/20-23:16:53.537037 7f179d7eb6c0 Manual compaction at level-1 from '!folders!7j8H7DbmBb9Uza2X' @ 72057594037927935 : 1 .. '!items!zt8s7564ep1La4XQ' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-skills/MANIFEST-000613 b/packs-system/lf-skills/MANIFEST-000617 similarity index 71% rename from packs-system/lf-skills/MANIFEST-000613 rename to packs-system/lf-skills/MANIFEST-000617 index 51f24ac68ae6132763e88c440349b156e1140c2d..e3b0a89dd8970d0ee9345a3cb86e9d2691686ff2 100644 GIT binary patch delta 43 tcmdnQxQTH>p9-f_?NM$9My6LRoD9sbSy&#jYM*($10?bqB=Qy{0stPB42=K) delta 43 tcmdnQxQTH>pUUS0x)R(Bj7-m1I2o9qv#>m5?Y}zh2T0^ONaQ6*1OPQt4M+e0 diff --git a/packs-system/lf-spells-miracles/000310.log b/packs-system/lf-spells-miracles/000314.log similarity index 100% rename from packs-system/lf-spells-miracles/000310.log rename to packs-system/lf-spells-miracles/000314.log diff --git a/packs-system/lf-spells-miracles/CURRENT b/packs-system/lf-spells-miracles/CURRENT index 43585a4..16e7795 100644 --- a/packs-system/lf-spells-miracles/CURRENT +++ b/packs-system/lf-spells-miracles/CURRENT @@ -1 +1 @@ -MANIFEST-000308 +MANIFEST-000312 diff --git a/packs-system/lf-spells-miracles/LOG b/packs-system/lf-spells-miracles/LOG index e566b1f..64ba27f 100644 --- a/packs-system/lf-spells-miracles/LOG +++ b/packs-system/lf-spells-miracles/LOG @@ -1,8 +1,8 @@ -2026/05/20-23:09:33.368596 7f179e7ed6c0 Recovering log #306 -2026/05/20-23:09:33.422152 7f179e7ed6c0 Delete type=3 #304 -2026/05/20-23:09:33.422210 7f179e7ed6c0 Delete type=0 #306 -2026/05/20-23:16:53.560903 7f179d7eb6c0 Level-0 table #311: started -2026/05/20-23:16:53.560969 7f179d7eb6c0 Level-0 table #311: 0 bytes OK -2026/05/20-23:16:53.568455 7f179d7eb6c0 Delete type=0 #309 -2026/05/20-23:16:53.568616 7f179d7eb6c0 Manual compaction at level-0 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end) -2026/05/20-23:16:53.578679 7f179d7eb6c0 Manual compaction at level-1 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end) +2026/05/23-18:45:10.252202 7fa2337fe6c0 Recovering log #310 +2026/05/23-18:45:10.261300 7fa2337fe6c0 Delete type=3 #308 +2026/05/23-18:45:10.261343 7fa2337fe6c0 Delete type=0 #310 +2026/05/23-19:09:56.467263 7fa2327fc6c0 Level-0 table #315: started +2026/05/23-19:09:56.467308 7fa2327fc6c0 Level-0 table #315: 0 bytes OK +2026/05/23-19:09:56.509084 7fa2327fc6c0 Delete type=0 #313 +2026/05/23-19:09:56.608544 7fa2327fc6c0 Manual compaction at level-0 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end) +2026/05/23-19:09:56.665018 7fa2327fc6c0 Manual compaction at level-1 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-spells-miracles/LOG.old b/packs-system/lf-spells-miracles/LOG.old index 47fcd32..e566b1f 100644 --- a/packs-system/lf-spells-miracles/LOG.old +++ b/packs-system/lf-spells-miracles/LOG.old @@ -1,8 +1,8 @@ -2026/05/20-00:10:37.136939 7f0b1efef6c0 Recovering log #302 -2026/05/20-00:10:37.146957 7f0b1efef6c0 Delete type=3 #300 -2026/05/20-00:10:37.147030 7f0b1efef6c0 Delete type=0 #302 -2026/05/20-10:53:40.615430 7f0b1cfeb6c0 Level-0 table #307: started -2026/05/20-10:53:40.615467 7f0b1cfeb6c0 Level-0 table #307: 0 bytes OK -2026/05/20-10:53:40.621586 7f0b1cfeb6c0 Delete type=0 #305 -2026/05/20-10:53:40.621713 7f0b1cfeb6c0 Manual compaction at level-0 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end) -2026/05/20-10:53:40.633723 7f0b1cfeb6c0 Manual compaction at level-1 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end) +2026/05/20-23:09:33.368596 7f179e7ed6c0 Recovering log #306 +2026/05/20-23:09:33.422152 7f179e7ed6c0 Delete type=3 #304 +2026/05/20-23:09:33.422210 7f179e7ed6c0 Delete type=0 #306 +2026/05/20-23:16:53.560903 7f179d7eb6c0 Level-0 table #311: started +2026/05/20-23:16:53.560969 7f179d7eb6c0 Level-0 table #311: 0 bytes OK +2026/05/20-23:16:53.568455 7f179d7eb6c0 Delete type=0 #309 +2026/05/20-23:16:53.568616 7f179d7eb6c0 Manual compaction at level-0 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end) +2026/05/20-23:16:53.578679 7f179d7eb6c0 Manual compaction at level-1 from '!folders!37mu4dxsSuftlnmP' @ 72057594037927935 : 1 .. '!items!zKOpU34oLziGJW6y' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-spells-miracles/MANIFEST-000308 b/packs-system/lf-spells-miracles/MANIFEST-000312 similarity index 72% rename from packs-system/lf-spells-miracles/MANIFEST-000308 rename to packs-system/lf-spells-miracles/MANIFEST-000312 index ee311e9415674a7a7e30eb1313064ef5175e0615..36dab678fe48a5abf1ff4449070316e4b702eddf 100644 GIT binary patch delta 43 tcmdnUxRG%}uS(w=)c|e=My8!ioD9snm{=xp=+<1c0g3DaiR=N1000aO3akJC delta 43 tcmdnUxRG%}uS!XF*<5Z0My9PyoD9s{m{=xpbhaL-0f}q_iR=K0000yz3o`%! diff --git a/packs-system/lf-vulnerabilities/000609.log b/packs-system/lf-vulnerabilities/000613.log similarity index 100% rename from packs-system/lf-vulnerabilities/000609.log rename to packs-system/lf-vulnerabilities/000613.log diff --git a/packs-system/lf-vulnerabilities/CURRENT b/packs-system/lf-vulnerabilities/CURRENT index 888567a..d0cc76d 100644 --- a/packs-system/lf-vulnerabilities/CURRENT +++ b/packs-system/lf-vulnerabilities/CURRENT @@ -1 +1 @@ -MANIFEST-000607 +MANIFEST-000611 diff --git a/packs-system/lf-vulnerabilities/LOG b/packs-system/lf-vulnerabilities/LOG index 2ea2d7e..46678bc 100644 --- a/packs-system/lf-vulnerabilities/LOG +++ b/packs-system/lf-vulnerabilities/LOG @@ -1,8 +1,8 @@ -2026/05/20-23:09:33.302872 7f179efee6c0 Recovering log #605 -2026/05/20-23:09:33.362229 7f179efee6c0 Delete type=3 #603 -2026/05/20-23:09:33.362319 7f179efee6c0 Delete type=0 #605 -2026/05/20-23:16:53.530498 7f179d7eb6c0 Level-0 table #610: started -2026/05/20-23:16:53.530547 7f179d7eb6c0 Level-0 table #610: 0 bytes OK -2026/05/20-23:16:53.536854 7f179d7eb6c0 Delete type=0 #608 -2026/05/20-23:16:53.537015 7f179d7eb6c0 Manual compaction at level-0 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end) -2026/05/20-23:16:53.537048 7f179d7eb6c0 Manual compaction at level-1 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end) +2026/05/23-18:45:10.238924 7fa233fff6c0 Recovering log #609 +2026/05/23-18:45:10.250309 7fa233fff6c0 Delete type=3 #607 +2026/05/23-18:45:10.250371 7fa233fff6c0 Delete type=0 #609 +2026/05/23-19:09:56.224507 7fa2327fc6c0 Level-0 table #614: started +2026/05/23-19:09:56.224539 7fa2327fc6c0 Level-0 table #614: 0 bytes OK +2026/05/23-19:09:56.261516 7fa2327fc6c0 Delete type=0 #612 +2026/05/23-19:09:56.339172 7fa2327fc6c0 Manual compaction at level-0 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end) +2026/05/23-19:09:56.339199 7fa2327fc6c0 Manual compaction at level-1 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-vulnerabilities/LOG.old b/packs-system/lf-vulnerabilities/LOG.old index de55e2b..2ea2d7e 100644 --- a/packs-system/lf-vulnerabilities/LOG.old +++ b/packs-system/lf-vulnerabilities/LOG.old @@ -1,8 +1,8 @@ -2026/05/20-00:10:37.121340 7f0b1e7ee6c0 Recovering log #601 -2026/05/20-00:10:37.132435 7f0b1e7ee6c0 Delete type=3 #599 -2026/05/20-00:10:37.132491 7f0b1e7ee6c0 Delete type=0 #601 -2026/05/20-10:53:40.559937 7f0b1cfeb6c0 Level-0 table #606: started -2026/05/20-10:53:40.560028 7f0b1cfeb6c0 Level-0 table #606: 0 bytes OK -2026/05/20-10:53:40.566605 7f0b1cfeb6c0 Delete type=0 #604 -2026/05/20-10:53:40.587338 7f0b1cfeb6c0 Manual compaction at level-0 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end) -2026/05/20-10:53:40.587398 7f0b1cfeb6c0 Manual compaction at level-1 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end) +2026/05/20-23:09:33.302872 7f179efee6c0 Recovering log #605 +2026/05/20-23:09:33.362229 7f179efee6c0 Delete type=3 #603 +2026/05/20-23:09:33.362319 7f179efee6c0 Delete type=0 #605 +2026/05/20-23:16:53.530498 7f179d7eb6c0 Level-0 table #610: started +2026/05/20-23:16:53.530547 7f179d7eb6c0 Level-0 table #610: 0 bytes OK +2026/05/20-23:16:53.536854 7f179d7eb6c0 Delete type=0 #608 +2026/05/20-23:16:53.537015 7f179d7eb6c0 Manual compaction at level-0 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end) +2026/05/20-23:16:53.537048 7f179d7eb6c0 Manual compaction at level-1 from '!folders!mnO9OzE7BEE2KDfh' @ 72057594037927935 : 1 .. '!items!zkK6ixtCsCw3RH9X' @ 0 : 0; will stop at (end) diff --git a/packs-system/lf-vulnerabilities/MANIFEST-000607 b/packs-system/lf-vulnerabilities/MANIFEST-000611 similarity index 72% rename from packs-system/lf-vulnerabilities/MANIFEST-000607 rename to packs-system/lf-vulnerabilities/MANIFEST-000611 index f552c0a501c0b1bbd086790fccb56755d9862c10..cec5e6c727894e9978d7a29b3dd8afa03c3174ff 100644 GIT binary patch delta 41 scmdnMxPfs(pOWgCq + +
+ {{localize "LETHALFANTASY.Label.rangeAttackDialog"}} + +
+ Attacker Movement : + +
+ +
+ Range : + +
+ +
+ Target Size : + +
+ +
+ Aim : + +
+ +
+ +
+ {{localize "LETHALFANTASY.Roll.visibility"}} + + + +
+ +