import DHBaseItemSheet from '../api/base-item.mjs'; const { TextEditor } = foundry.applications.ux; export default class ClassSheet extends DHBaseItemSheet { /**@inheritdoc */ static DEFAULT_OPTIONS = { classes: ['class'], position: { width: 700 }, actions: { removeItemFromCollection: ClassSheet.#removeItemFromCollection, removeSuggestedItem: ClassSheet.#removeSuggestedItem }, tagifyConfigs: [ { selector: '.domain-input', options: () => CONFIG.DH.DOMAIN.domains, callback: ClassSheet.#onDomainSelect } ], dragDrop: [ { dragSelector: '.suggested-item', dropSelector: null }, { dragSelector: null, dropSelector: '.take-section' }, { dragSelector: null, dropSelector: '.choice-a-section' }, { dragSelector: null, dropSelector: '.choice-b-section' }, { dragSelector: null, dropSelector: '.primary-weapon-section' }, { dragSelector: null, dropSelector: '.secondary-weapon-section' }, { dragSelector: null, dropSelector: '.armor-section' }, { dragSelector: null, dropSelector: null } ] }; /**@override */ static PARTS = { header: { template: 'systems/daggerheart/templates/sheets/items/class/header.hbs' }, tabs: { template: 'systems/daggerheart/templates/sheets/global/tabs/tab-navigation.hbs' }, description: { template: 'systems/daggerheart/templates/sheets/global/tabs/tab-description.hbs' }, features: { template: 'systems/daggerheart/templates/sheets/items/class/features.hbs', scrollable: ['.features'] }, settings: { template: 'systems/daggerheart/templates/sheets/items/class/settings.hbs', scrollable: ['.settings'] }, effects: { template: 'systems/daggerheart/templates/sheets/global/tabs/tab-effects.hbs', scrollable: ['.effects'] } }; /** @inheritdoc */ static TABS = { primary: { tabs: [{ id: 'description' }, { id: 'features' }, { id: 'settings' }, { id: 'effects' }], initial: 'description', labelPrefix: 'DAGGERHEART.GENERAL.Tabs' } }; /**@inheritdoc */ get relatedDocs() { return this.document.system.features.map(x => x.item); } /**@inheritdoc */ async _onFirstRender(context, options) { await super._onFirstRender(context, options); const paths = [ 'subclasses', 'characterGuide.suggestedPrimaryWeapon', 'characterGuide.suggestedSecondaryWeapon', 'characterGuide.suggestedArmor', 'inventory.take', 'inventory.choiceA', 'inventory.choiceB' ]; paths.forEach(path => { const docs = [].concat(foundry.utils.getProperty(this.document, `system.${path}`) ?? []); docs.forEach(doc => (doc.apps[this.id] = this)); }); } /**@inheritdoc */ async _prepareContext(_options) { const context = await super._prepareContext(_options); context.domains = this.document.system.domains; return context; } /* -------------------------------------------- */ /** * Callback function used by `tagifyElement`. * @param {Array} selectedOptions - The currently selected tag objects. */ static async #onDomainSelect(selectedOptions) { await this.document.update({ 'system.domains': selectedOptions.map(x => x.value) }); } /* -------------------------------------------- */ async _onDrop(event) { event.stopPropagation(); const data = TextEditor.getDragEventData(event); const item = await fromUuid(data.uuid); const target = event.target.closest('fieldset.drop-section'); if (item.type === 'subclass') { await this.document.update({ 'system.subclasses': [...this.document.system.subclasses.map(x => x.uuid), item.uuid] }); } else if (item.type === 'feature') { super._onDrop(event); } else if (this.document.parent?.type !== 'character') { if (item.type === 'weapon') { if (target.classList.contains('primary-weapon-section')) { if (!item.system.secondary) await this.document.update({ 'system.characterGuide.suggestedPrimaryWeapon': item.uuid }); } else if (target.classList.contains('secondary-weapon-section')) { if (item.system.secondary) await this.document.update({ 'system.characterGuide.suggestedSecondaryWeapon': item.uuid }); } } else if (item.type === 'armor') { if (target.classList.contains('armor-section')) { await this.document.update({ 'system.characterGuide.suggestedArmor': item.uuid }); } } else if (target.classList.contains('choice-a-section')) { if (item.type === 'loot' || item.type === 'consumable') { const filteredChoiceA = this.document.system.inventory.choiceA; if (filteredChoiceA.length < 2) await this.document.update({ 'system.inventory.choiceA': [...filteredChoiceA.map(x => x.uuid), item.uuid] }); } } else if (item.type === 'loot') { if (target.classList.contains('take-section')) { const filteredTake = this.document.system.inventory.take.filter(x => x); if (filteredTake.length < 3) await this.document.update({ 'system.inventory.take': [...filteredTake.map(x => x.uuid), item.uuid] }); } else if (target.classList.contains('choice-b-section')) { const filteredChoiceB = this.document.system.inventory.choiceB.filter(x => x); if (filteredChoiceB.length < 2) await this.document.update({ 'system.inventory.choiceB': [...filteredChoiceB.map(x => x.uuid), item.uuid] }); } } } } /* -------------------------------------------- */ /* Application Clicks Actions */ /* -------------------------------------------- */ /** * Removes an item from an class collection by UUID. * @param {PointerEvent} event - The originating click event * @param {HTMLElement} element - The capturing HTML element which defines the [data-action="removeItemFromCollection"] */ static async #removeItemFromCollection(_event, element) { const { uuid, target } = element.dataset; const prop = foundry.utils.getProperty(this.document.system, target); await this.document.update({ [`system.${target}`]: prop.filter(i => i.uuid !== uuid).map(x => x.uuid) }); } /** * Removes an suggested item from the class. * @param {PointerEvent} _event - The originating click event * @param {HTMLElement} element - The capturing HTML element which defines the [data-action="removeSuggestedItem"] */ static async #removeSuggestedItem(_event, element) { const { target } = element.dataset; await this.document.update({ [`system.characterGuide.${target}`]: null }); } }