From cbac5873dc648f3c5ba82e4e25014446e4a62782 Mon Sep 17 00:00:00 2001 From: Jeff Hitchcock Date: Sun, 7 Aug 2022 10:49:23 -0700 Subject: [PATCH 1/4] [#1620, #1626, #1689] Add spell config & type restriction to ItemGrant advancement --- lang/en.json | 1 + module/advancement/advancement-config.mjs | 101 +++++++++++++++++- module/advancement/advancement.mjs | 27 +++++ module/advancement/types/item-grant.mjs | 81 +++++--------- module/utils.mjs | 3 +- templates/advancement/item-grant-config.hbs | 8 +- .../parts/advancement-spell-config.hbs | 28 +++++ 7 files changed, 189 insertions(+), 60 deletions(-) create mode 100644 templates/advancement/parts/advancement-spell-config.hbs diff --git a/lang/en.json b/lang/en.json index 6b4fcca69f..0d35a2ce18 100644 --- a/lang/en.json +++ b/lang/en.json @@ -118,6 +118,7 @@ "DND5E.AdvancementItemGrantRecursiveWarning": "You cannot grant an item in its own advancement.", "DND5E.AdvancementItemGrantOptional": "Optional", "DND5E.AdvancementItemGrantOptionalHint": "If optional, players will be given the option to opt out of any of the following items, otherwise all of them are granted.", +"DND5E.AdvancementItemTypeInvalidWarning": "{type} items cannot be added with this advancement type.", "DND5E.AdvancementLevelHeader": "Level {level}", "DND5E.AdvancementLevelAnyHeader": "Any Level", "DND5E.AdvancementLevelNoneHeader": "No Level", diff --git a/module/advancement/advancement-config.mjs b/module/advancement/advancement-config.mjs index bdd95d7a25..8211ca61ba 100644 --- a/module/advancement/advancement-config.mjs +++ b/module/advancement/advancement-config.mjs @@ -2,8 +2,10 @@ * Base configuration application for advancements that can be extended by other types to implement custom * editing interfaces. * - * @param {Advancement} advancement The advancement item being edited. - * @param {object} [options={}] Additional options passed to FormApplication. + * @property {Advancement} advancement The advancement item being edited. + * @param {object} [options={}] Additional options passed to FormApplication. + * @param {string} [options.dropKeyPath=null] Path within advancement configuration where dropped items are stored. + * If populated, will enable default drop & delete behavior. */ export default class AdvancementConfig extends FormApplication { constructor(advancement, options={}) { @@ -32,7 +34,8 @@ export default class AdvancementConfig extends FormApplication { width: 400, height: "auto", submitOnChange: true, - closeOnSubmit: false + closeOnSubmit: false, + dropKeyPath: null }); } @@ -52,6 +55,7 @@ export default class AdvancementConfig extends FormApplication { if ( ["class", "subclass"].includes(this.item.type) ) delete levels[0]; else levels[0] = game.i18n.localize("DND5E.AdvancementLevelAnyHeader"); return { + CONFIG: CONFIG.DND5E, data: this.advancement.data, default: { title: this.advancement.constructor.metadata.title, @@ -76,6 +80,15 @@ export default class AdvancementConfig extends FormApplication { /* -------------------------------------------- */ + activateListeners(html) { + super.activateListeners(html); + + // Remove an item from the list + if ( this.options.dropKeyPath ) html.on("click", ".item-delete", this._onItemDelete.bind(this)); + } + + /* -------------------------------------------- */ + /** @inheritdoc */ async _updateObject(event, formData) { let updates = foundry.utils.expandObject(formData).data; @@ -101,4 +114,86 @@ export default class AdvancementConfig extends FormApplication { }, {}); } + /* -------------------------------------------- */ + /* Drag & Drop for Item Pools */ + /* -------------------------------------------- */ + + /** + * Handle deleting an existing Item entry from the Advancement. + * @param {Event} event The originating click event. + * @returns {Promise} The updated parent Item after the application re-renders. + * @protected + */ + async _onItemDelete(event) { + event.preventDefault(); + const uuidToDelete = event.currentTarget.closest("[data-item-uuid]")?.dataset.itemUuid; + if ( !uuidToDelete ) return; + const items = foundry.utils.getProperty(this.advancement.data.configuration, this.options.dropKeyPath); + const updates = { configuration: await this.prepareConfigurationUpdate({ + [this.options.dropKeyPath]: items.filter(uuid => uuid !== uuidToDelete) + }) }; + console.log(updates); + await this.advancement.update(updates); + this.render(); + } + + /* -------------------------------------------- */ + + /** @inheritdoc */ + _canDragDrop() { + return this.isEditable; + } + + /* -------------------------------------------- */ + + /** @inheritdoc */ + async _onDrop(event) { + if ( !this.options.dropKeyPath ) return console.error( + "AdvancementConfig#options.dropKeyPath must be configured or #_onDrop must be overridden to support" + + " drag and drop on advancement config items." + ); + + // Try to extract the data + let data; + try { + data = JSON.parse(event.dataTransfer.getData("text/plain")); + } catch(err) { + return false; + } + + if ( data.type !== "Item" ) return false; + const item = await Item.implementation.fromDropData(data); + + const verified = this._verifyDroppedItem(event, item); + if ( !verified ) return false; + + const existingItems = foundry.utils.getProperty(this.advancement.data.configuration, this.options.dropKeyPath); + + // Abort if this uuid is the parent item + if ( item.uuid === this.item.uuid ) { + return ui.notifications.warn(game.i18n.localize("DND5E.AdvancementItemGrantRecursiveWarning")); + } + + // Abort if this uuid exists already + if ( existingItems.includes(item.uuid) ) { + return ui.notifications.warn(game.i18n.localize("DND5E.AdvancementItemGrantDuplicateWarning")); + } + + await this.advancement.update({[`configuration.${this.options.dropKeyPath}`]: [...existingItems, item.uuid]}); + this.render(); + } + + /* -------------------------------------------- */ + + /** + * Called when an item is dropped to verify the Item before it is saved. + * @param {Event} event Triggering drop event. + * @param {Item5e} item The materialized Item that was dropped. + * @returns {boolean} Is the dropped Item valid? + * @protected + */ + _verifyDroppedItem(event, item) { + return true; + } + } diff --git a/module/advancement/advancement.mjs b/module/advancement/advancement.mjs index 35b78440a5..18f7be5b79 100644 --- a/module/advancement/advancement.mjs +++ b/module/advancement/advancement.mjs @@ -318,4 +318,31 @@ export default class Advancement { * @abstract */ async reverse(level) { } + + /* -------------------------------------------- */ + + /** + * Helper method to prepare spell customizations. + * @param {object} spell Spell configuration object. + * @returns {object} Object of updates to apply to item. + * @protected + */ + _prepareSpellChanges(spell) { + const updates = {}; + if ( spell.ability ) updates["system.ability"] = spell.ability; + if ( spell.preparation ) updates["system.preparation.mode"] = spell.preparation; + if ( spell.uses?.max ) { + updates["system.uses.max"] = spell.uses.max; + if ( Number.isNumeric(spell.uses.max) ) updates["system.uses.value"] = parseInt(spell.uses.max); + else { + try { + const rollData = this.actor.getRollData({ deterministic: true }); + const formula = Roll.replaceFormulaData(spell.uses.max, rollData, {missing: 0}); + updates["system.uses.value"] = Roll.safeEval(formula); + } catch(e) { } + } + } + if ( spell.uses?.per ) updates["system.uses.per"] = spell.uses.per; + return updates; + } } diff --git a/module/advancement/types/item-grant.mjs b/module/advancement/types/item-grant.mjs index 597d02acd4..dec3b6158b 100644 --- a/module/advancement/types/item-grant.mjs +++ b/module/advancement/types/item-grant.mjs @@ -12,7 +12,11 @@ export class ItemGrantAdvancement extends Advancement { static get metadata() { return foundry.utils.mergeObject(super.metadata, { defaults: { - configuration: { items: [], optional: false } + configuration: { + items: [], + optional: false, + spell: null + } }, order: 40, icon: "systems/dnd5e/icons/svg/item-grant.svg", @@ -25,6 +29,14 @@ export class ItemGrantAdvancement extends Advancement { }); } + /* -------------------------------------------- */ + + /** + * The item types that are supported in Item Grant. + * @type {Set} + */ + static VALID_TYPES = new Set(["feat", "spell", "consumable", "backpack", "equipment", "loot", "tool", "weapon"]); + /* -------------------------------------------- */ /* Display Methods */ /* -------------------------------------------- */ @@ -66,6 +78,7 @@ export class ItemGrantAdvancement extends Advancement { async apply(level, data, retainedData={}) { const items = []; const updates = {}; + const spellChanges = this.data.configuration.spell ? this._prepareSpellChanges(this.data.configuration.spell) : {}; for ( const [uuid, selected] of Object.entries(data) ) { if ( !selected ) continue; @@ -79,6 +92,7 @@ export class ItemGrantAdvancement extends Advancement { "flags.dnd5e.advancementOrigin": `${this.item.id}.${this.id}` }, {keepId: true}).toObject(); } + foundry.utils.mergeObject(itemData, spellChanges); items.push(itemData); // TODO: Trigger any additional advancement steps for added items @@ -129,72 +143,29 @@ export class ItemGrantConfig extends AdvancementConfig { static get defaultOptions() { return foundry.utils.mergeObject(super.defaultOptions, { dragDrop: [{ dropSelector: ".drop-target" }], + dropKeyPath: "items", template: "systems/dnd5e/templates/advancement/item-grant-config.hbs" }); } /* -------------------------------------------- */ - activateListeners(html) { - super.activateListeners(html); - - // Remove an item from the list - html.on("click", ".item-delete", this._onItemDelete.bind(this)); - } - - /* -------------------------------------------- */ - - /** - * Handle deleting an existing Item entry from the Advancement. - * @param {Event} event The originating click event. - * @returns {Promise} The promise for the updated parent Item which resolves after the application re-renders - * @private - */ - async _onItemDelete(event) { - event.preventDefault(); - const uuidToDelete = event.currentTarget.closest("[data-item-uuid]")?.dataset.itemUuid; - if ( !uuidToDelete ) return; - const items = this.advancement.data.configuration.items.filter(uuid => uuid !== uuidToDelete); - const updates = { configuration: this.prepareConfigurationUpdate({ items }) }; - await this.advancement.update(updates); - this.render(); - } - - /* -------------------------------------------- */ - /** @inheritdoc */ - _canDragDrop() { - return this.isEditable; + async getData() { + const data = super.getData(); + data.items = data.data.configuration.items.map(fromUuidSync); + data.showSpellConfig = data.items.some(i => i.type === "spell"); + return data; } /* -------------------------------------------- */ /** @inheritdoc */ - async _onDrop(event) { - // Try to extract the data - let data; - try { - data = JSON.parse(event.dataTransfer.getData("text/plain")); - } catch(err) { - return false; - } - - if ( data.type !== "Item" ) return false; - const item = await Item.implementation.fromDropData(data); - const existingItems = this.advancement.data.configuration.items; - - // Abort if this uuid is the parent item - if ( item.uuid === this.item.uuid ) { - return ui.notifications.warn(game.i18n.localize("DND5E.AdvancementItemGrantRecursiveWarning")); - } - - // Abort if this uuid exists already - if ( existingItems.includes(item.uuid) ) { - return ui.notifications.warn(game.i18n.localize("DND5E.AdvancementItemGrantDuplicateWarning")); - } - - await this.advancement.update({"configuration.items": [...existingItems, item.uuid]}); - this.render(); + _verifyDroppedItem(event, item) { + if ( this.advancement.constructor.VALID_TYPES.has(item.type) ) return true; + const type = game.i18n.localize(`ITEM.Type${item.type.capitalize()}`); + ui.notifications.warn(game.i18n.format("DND5E.AdvancementItemTypeInvalidWarning", { type })); + return false; } } diff --git a/module/utils.mjs b/module/utils.mjs index b56468e257..22480cefd1 100644 --- a/module/utils.mjs +++ b/module/utils.mjs @@ -110,7 +110,8 @@ export async function preloadHandlebarsTemplates() { "systems/dnd5e/templates/items/parts/item-spellcasting.hbs", // Advancement Partials - "systems/dnd5e/templates/advancement/parts/advancement-controls.hbs" + "systems/dnd5e/templates/advancement/parts/advancement-controls.hbs", + "systems/dnd5e/templates/advancement/parts/advancement-spell-config.hbs" ]; const paths = {}; diff --git a/templates/advancement/item-grant-config.hbs b/templates/advancement/item-grant-config.hbs index 8b78ca2b45..abf5a08216 100644 --- a/templates/advancement/item-grant-config.hbs +++ b/templates/advancement/item-grant-config.hbs @@ -8,6 +8,10 @@

{{localize "DND5E.AdvancementItemGrantOptionalHint"}}

+ {{#if showSpellConfig}} + {{> "dnd5e.advancement-spell-config"}} + {{/if}} +
  1. {{localize "DOCUMENT.Items"}}

  2. @@ -16,7 +20,9 @@
  3. {{{dnd5e-linkForUuid this}}}
    - + + +
  4. {{/each}} diff --git a/templates/advancement/parts/advancement-spell-config.hbs b/templates/advancement/parts/advancement-spell-config.hbs new file mode 100644 index 0000000000..1abf2c2bfe --- /dev/null +++ b/templates/advancement/parts/advancement-spell-config.hbs @@ -0,0 +1,28 @@ +
    + {{log this}} + +
    + +
    +
    + +
    + +
    + +
    +
    + +
    + +
    + + +
    +
    From 5597ddf13e5cb627615798afce843741a36aa61f Mon Sep 17 00:00:00 2001 From: Jeff Hitchcock Date: Sun, 7 Aug 2022 10:56:23 -0700 Subject: [PATCH 2/4] [#1626] Fix issue with applying spell changes to non-spell items --- module/advancement/types/item-grant.mjs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/module/advancement/types/item-grant.mjs b/module/advancement/types/item-grant.mjs index dec3b6158b..acd648904b 100644 --- a/module/advancement/types/item-grant.mjs +++ b/module/advancement/types/item-grant.mjs @@ -92,7 +92,7 @@ export class ItemGrantAdvancement extends Advancement { "flags.dnd5e.advancementOrigin": `${this.item.id}.${this.id}` }, {keepId: true}).toObject(); } - foundry.utils.mergeObject(itemData, spellChanges); + if ( itemData.type === "spell" ) foundry.utils.mergeObject(itemData, spellChanges); items.push(itemData); // TODO: Trigger any additional advancement steps for added items From c17285cdd9961e6ea23848a6b20e090034d2d237 Mon Sep 17 00:00:00 2001 From: Jeff Hitchcock Date: Sat, 27 Aug 2022 08:51:04 -0700 Subject: [PATCH 3/4] [#1626, 1689] Adjust how some drop errors are handled, remove development logging --- module/advancement/advancement-config.mjs | 22 ++++++++++----------- module/advancement/types/item-grant.mjs | 5 ++--- templates/advancement/item-grant-config.hbs | 2 +- 3 files changed, 14 insertions(+), 15 deletions(-) diff --git a/module/advancement/advancement-config.mjs b/module/advancement/advancement-config.mjs index 8211ca61ba..c163524168 100644 --- a/module/advancement/advancement-config.mjs +++ b/module/advancement/advancement-config.mjs @@ -84,7 +84,7 @@ export default class AdvancementConfig extends FormApplication { super.activateListeners(html); // Remove an item from the list - if ( this.options.dropKeyPath ) html.on("click", ".item-delete", this._onItemDelete.bind(this)); + if ( this.options.dropKeyPath ) html.on("click", "[data-action='delete']", this._onItemDelete.bind(this)); } /* -------------------------------------------- */ @@ -132,7 +132,6 @@ export default class AdvancementConfig extends FormApplication { const updates = { configuration: await this.prepareConfigurationUpdate({ [this.options.dropKeyPath]: items.filter(uuid => uuid !== uuidToDelete) }) }; - console.log(updates); await this.advancement.update(updates); this.render(); } @@ -148,7 +147,7 @@ export default class AdvancementConfig extends FormApplication { /** @inheritdoc */ async _onDrop(event) { - if ( !this.options.dropKeyPath ) return console.error( + if ( !this.options.dropKeyPath ) throw new Error( "AdvancementConfig#options.dropKeyPath must be configured or #_onDrop must be overridden to support" + " drag and drop on advancement config items." ); @@ -164,14 +163,17 @@ export default class AdvancementConfig extends FormApplication { if ( data.type !== "Item" ) return false; const item = await Item.implementation.fromDropData(data); - const verified = this._verifyDroppedItem(event, item); - if ( !verified ) return false; + try { + this._validateDroppedItem(event, item); + } catch(err) { + return ui.notifications.error(err.message); + } const existingItems = foundry.utils.getProperty(this.advancement.data.configuration, this.options.dropKeyPath); // Abort if this uuid is the parent item if ( item.uuid === this.item.uuid ) { - return ui.notifications.warn(game.i18n.localize("DND5E.AdvancementItemGrantRecursiveWarning")); + return ui.notifications.error(game.i18n.localize("DND5E.AdvancementItemGrantRecursiveWarning")); } // Abort if this uuid exists already @@ -186,14 +188,12 @@ export default class AdvancementConfig extends FormApplication { /* -------------------------------------------- */ /** - * Called when an item is dropped to verify the Item before it is saved. + * Called when an item is dropped to validate the Item before it is saved. An error should be thrown + * if the item is invalid. * @param {Event} event Triggering drop event. * @param {Item5e} item The materialized Item that was dropped. - * @returns {boolean} Is the dropped Item valid? * @protected */ - _verifyDroppedItem(event, item) { - return true; - } + _validateDroppedItem(event, item) {} } diff --git a/module/advancement/types/item-grant.mjs b/module/advancement/types/item-grant.mjs index acd648904b..1e28d36340 100644 --- a/module/advancement/types/item-grant.mjs +++ b/module/advancement/types/item-grant.mjs @@ -161,11 +161,10 @@ export class ItemGrantConfig extends AdvancementConfig { /* -------------------------------------------- */ /** @inheritdoc */ - _verifyDroppedItem(event, item) { + _validateDroppedItem(event, item) { if ( this.advancement.constructor.VALID_TYPES.has(item.type) ) return true; const type = game.i18n.localize(`ITEM.Type${item.type.capitalize()}`); - ui.notifications.warn(game.i18n.format("DND5E.AdvancementItemTypeInvalidWarning", { type })); - return false; + throw new Error(game.i18n.format("DND5E.AdvancementItemTypeInvalidWarning", { type })); } } diff --git a/templates/advancement/item-grant-config.hbs b/templates/advancement/item-grant-config.hbs index abf5a08216..f520ade670 100644 --- a/templates/advancement/item-grant-config.hbs +++ b/templates/advancement/item-grant-config.hbs @@ -20,7 +20,7 @@
  5. {{{dnd5e-linkForUuid this}}}
    From bd31df1ad9bc94a5805391cdf6a73b40cfc24b1f Mon Sep 17 00:00:00 2001 From: Jeff Hitchcock Date: Fri, 9 Sep 2022 12:10:31 -0700 Subject: [PATCH 4/4] [#1626, #1689] Add throws annotation to _validateDroppedItem, remove async requirement from ItemGrantConfig#getData --- module/advancement/advancement-config.mjs | 1 + module/advancement/types/item-grant.mjs | 9 ++++----- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/module/advancement/advancement-config.mjs b/module/advancement/advancement-config.mjs index c163524168..ef71cdb020 100644 --- a/module/advancement/advancement-config.mjs +++ b/module/advancement/advancement-config.mjs @@ -192,6 +192,7 @@ export default class AdvancementConfig extends FormApplication { * if the item is invalid. * @param {Event} event Triggering drop event. * @param {Item5e} item The materialized Item that was dropped. + * @throws An error if the item is invalid. * @protected */ _validateDroppedItem(event, item) {} diff --git a/module/advancement/types/item-grant.mjs b/module/advancement/types/item-grant.mjs index 1e28d36340..367982740c 100644 --- a/module/advancement/types/item-grant.mjs +++ b/module/advancement/types/item-grant.mjs @@ -151,11 +151,10 @@ export class ItemGrantConfig extends AdvancementConfig { /* -------------------------------------------- */ /** @inheritdoc */ - async getData() { - const data = super.getData(); - data.items = data.data.configuration.items.map(fromUuidSync); - data.showSpellConfig = data.items.some(i => i.type === "spell"); - return data; + getData() { + const context = super.getData(); + context.showSpellConfig = context.data.configuration.items.map(fromUuidSync).some(i => i.type === "spell"); + return context; } /* -------------------------------------------- */