diff --git a/src/lang/de.json b/src/lang/de.json index 11b879c..e9b8888 100644 --- a/src/lang/de.json +++ b/src/lang/de.json @@ -189,6 +189,8 @@ "DS4.ErrorRollingForItemTypeNotPossible": "Würfeln ist für Items vom Typ '{type}' nicht möglich.", "DS4.ErrorWrongItemType": "Ein Item vom Type '{expectedType}' wurde erwartet aber das Item '{name}' ({id}) ist vom Typ '{actualType}'.", "DS4.ErrorUnexpectedAttackType": "Unerwartete Angriffsart '{actualType}', erwartete Angriffarten: {expectedTypes}", + "DS4.ErrorItemMustBeEquippedToBeRolled": "Um für das Item '{name}' ({id}) vom Typ {type} zu würfeln, muss es ausgerüstet sein.", + "DS4.InfoManuallyEnterSpellBonus": "Der korrekte Wert für den Zauberbonus '{spellBonus}' des Zaubers '{name}' musss manuell angegeben werden.", "DS4.InfoSystemUpdateStart": "Aktualisiere DS4 System von Migrationsversion {currentVersion} auf {targetVersion}. Bitte haben Sie etwas Geduld, schließen Sie nicht das Spiel und fahren Sie nicht den Server herunter.", "DS4.InfoSystemUpdateCompleted": "Aktualisierung des DS4 Systems von Migrationsversion {currentVersion} auf {targetVersion} erfolgreich!", "DS4.UnitRounds": "Runden", diff --git a/src/lang/en.json b/src/lang/en.json index f1436c4..ec17658 100644 --- a/src/lang/en.json +++ b/src/lang/en.json @@ -189,6 +189,8 @@ "DS4.ErrorRollingForItemTypeNotPossible": "Rolling is not possible for items of type '{type}'.", "DS4.ErrorWrongItemType": "Expected an item of type '{expectedType}' but item '{name}' ({id}) is of type '{actualType}'.", "DS4.ErrorUnexpectedAttackType": "Unexpected attack type '{actualType}', expected it to be one of: {expectedTypes}", + "DS4.ErrorItemMustBeEquippedToBeRolled": "To roll for item '{name}' ({id}) of type {type}, it needs to be equipped.", + "DS4.InfoManuallyEnterSpellBonus": "The correct value of the spell bons '{spellBonus}' of the spell '{name}' needs to be entered by manually.", "DS4.InfoSystemUpdateStart": "Migrating DS4 system from migration version {currentVersion} to {targetVersion}. Please be patient and do not close your game or shut down your server.", "DS4.InfoSystemUpdateCompleted": "Migration of DS4 system from migration version {currentVersion} to {targetVersion} successful!", "DS4.UnitRounds": "Rounds", diff --git a/src/module/item/item-data.ts b/src/module/item/item-data.ts index c3cf9a9..c28fbe8 100644 --- a/src/module/item/item-data.ts +++ b/src/module/item/item-data.ts @@ -34,11 +34,14 @@ type DS4SpecialCreatureAbilityData = DS4ItemDataHelper { max: number; } -interface DS4SpellDataData extends DS4ItemDataDataBase, DS4ItemDataDataEquipable { +interface DS4SpellDataData extends DS4ItemDataDataBase, DS4ItemDataDataEquipable, DS4ItemDataDataRollable { spellType: "spellcasting" | "targetedSpellcasting"; bonus: string; spellCategory: @@ -112,6 +115,10 @@ interface DS4ItemDataDataEquipable { equipped: boolean; } +interface DS4ItemDataDataRollable { + rollable?: boolean; +} + interface DS4ItemDataDataProtective { armorValue: number; } diff --git a/src/module/item/item.ts b/src/module/item/item.ts index 211f772..52d5941 100644 --- a/src/module/item/item.ts +++ b/src/module/item/item.ts @@ -1,6 +1,7 @@ import { DS4Actor } from "../actor/actor"; import { DS4 } from "../config"; import { createCheckRoll } from "../rolls/check-factory"; +import notifications from "../ui/notifications"; import { AttackType, DS4ItemData } from "./item-data"; /** @@ -20,8 +21,8 @@ export class DS4Item extends Item { const data = this.data.data; data.rank.total = data.rank.base + data.rank.mod; } - if (this.data.type === "weapon") { - this.data.data.rollable = true; + if (this.data.type === "weapon" || this.data.type === "spell") { + this.data.data.rollable = this.data.data.equipped; } } @@ -46,10 +47,14 @@ export class DS4Item extends Item { if (!this.isOwnedItem()) { throw new Error(game.i18n.format("DS4.ErrorCannotRollUnownedItem", { name: this.name, id: this.id })); } - if (this.data.type === "weapon") { - await this.rollWeapon(); - } else { - throw new Error(game.i18n.format("DS4.ErrorRollingForItemTypeNotPossible", { type: this.data.type })); + + switch (this.data.type) { + case "weapon": + await this.rollWeapon(); + case "spell": + await this.rollSpell(); + default: + throw new Error(game.i18n.format("DS4.ErrorRollingForItemTypeNotPossible", { type: this.data.type })); } } @@ -65,6 +70,16 @@ export class DS4Item extends Item { ); } + if (!this.data.data.equipped) { + throw new Error( + game.i18n.format("DS4.ErrorItemMustBeEquippedToBeRolled", { + name: this.name, + id: this.id, + type: this.data.type, + }), + ); + } + const ownerDataData = ((this.actor as unknown) as DS4Actor).data.data; // TODO(types): Improve so that the concrete Actor type is known here const weaponBonus = this.data.data.weaponBonus; const combatValue = await this.getCombatValueKeyForAttackType(this.data.data.attackType); @@ -76,6 +91,48 @@ export class DS4Item extends Item { }); } + private async rollSpell(): Promise { + if (!(this.data.type === "spell")) { + throw new Error( + game.i18n.format("DS4.ErrorWrongItemType", { + actualType: this.data.type, + expectedType: "spell", + id: this.id, + name: this.name, + }), + ); + } + + if (!this.data.data.equipped) { + throw new Error( + game.i18n.format("DS4.ErrorItemMustBeEquippedToBeRolled", { + name: this.name, + id: this.id, + type: this.data.type, + }), + ); + } + + const ownerDataData = ((this.actor as unknown) as DS4Actor).data.data; // TODO(types): Improve so that the concrete Actor type is known here + const spellBonus = Number.isNumeric(this.data.data.bonus) ? parseInt(this.data.data.bonus) : undefined; + if (spellBonus === undefined) { + notifications.info( + game.i18n.format("DS4.InfoManuallyEnterSpellBonus", { + name: this.name, + spellBonus: this.data.data.bonus, + }), + ); + } + const spellType = this.data.data.spellType; + const checkTargetValue = (ownerDataData.combatValues[spellType].total as number) + (spellBonus ?? 0); + + await createCheckRoll(checkTargetValue, { + rollMode: game.settings.get("core", "rollMode"), + maxCritSuccess: ownerDataData.rolling.maximumCoupResult, + minCritFailure: ownerDataData.rolling.minimumFumbleResult, + }); + } + private async getCombatValueKeyForAttackType(attackType: AttackType): Promise<"meleeAttack" | "rangedAttack"> { if (attackType === "meleeRanged") { const { melee, ranged } = { ...DS4.i18n.attackTypes };