mirror of
https://github.com/Foundryborne/daggerheart.git
synced 2026-01-12 11:41:08 +01:00
273 lines
11 KiB
JavaScript
273 lines
11 KiB
JavaScript
const { HandlebarsApplicationMixin } = foundry.applications.api;
|
|
import { tagifyElement } from '../../../helpers/utils.mjs';
|
|
|
|
/**
|
|
* @typedef {object} DragDropConfig
|
|
* @property {string} [dragSelector] - A CSS selector that identifies draggable elements.
|
|
* @property {string} [dropSelector] - A CSS selector that identifies drop targets.
|
|
*
|
|
* @typedef {object} ContextMenuConfig
|
|
* @property {() => ContextMenuEntry[]} handler - A handler function that provides initial context options
|
|
* @property {string} selector - A CSS selector to which the ContextMenu will be bound
|
|
* @property {object} [options] - Additional options which affect ContextMenu construction
|
|
* @property {HTMLElement} [options.container] - A parent HTMLElement which contains the selector target
|
|
* @property {string} [options.hookName] - The hook name
|
|
* @property {boolean} [options.parentClassHooks=true] - Whether to call hooks for the parent classes in the inheritance chain.
|
|
*
|
|
* @typedef {Object} TagOption
|
|
* @property {string} label
|
|
* @property {string} [src]
|
|
*
|
|
* @typedef {object} TagifyConfig
|
|
* @property {String} selector - The CSS selector for get the element to transform into a tag input
|
|
* @property {Record<string, TagOption> | (() => Record<string, TagOption>)} options - Available tag options as key-value pairs
|
|
* @property {TagChangeCallback} callback - Callback function triggered when tags change
|
|
* @property {TagifyOptions} [tagifyOptions={}] - Additional configuration for Tagify
|
|
*
|
|
* @callback TagChangeCallback
|
|
* @param {Array<{value: string, name: string, src?: string}>} selectedOptions - Current selected tags
|
|
* @param {{option: string, removed: boolean}} change - What changed (added/removed tag)
|
|
* @param {HTMLElement} inputElement - Original input element
|
|
*
|
|
*
|
|
* @typedef {Object} TagifyOptions
|
|
* @property {number} [maxTags] - Maximum number of allowed tags
|
|
*/
|
|
|
|
/**
|
|
* @typedef {import("@client/applications/api/handlebars-application.mjs").HandlebarsRenderOptions} HandlebarsRenderOptions
|
|
* @typedef {foundry.applications.types.ApplicationConfiguration} FoundryAppConfig
|
|
*
|
|
* @typedef {FoundryAppConfig & HandlebarsRenderOptions & {
|
|
* dragDrop?: DragDropConfig[],
|
|
* tagifyConfigs?: TagifyConfig[],
|
|
* contextMenus?: ContextMenuConfig[],
|
|
* }} DHSheetV2Configuration
|
|
*/
|
|
|
|
/**
|
|
* @template {Constructor<foundry.applications.api.DocumentSheet>} BaseDocumentSheet
|
|
* @param {BaseDocumentSheet} Base - The base class to extend.
|
|
* @returns {BaseDocumentSheet}
|
|
*/
|
|
export default function DHApplicationMixin(Base) {
|
|
class DHSheetV2 extends HandlebarsApplicationMixin(Base) {
|
|
/**
|
|
* @param {DHSheetV2Configuration} [options={}]
|
|
*/
|
|
constructor(options = {}) {
|
|
super(options);
|
|
/**
|
|
* @type {foundry.applications.ux.DragDrop[]}
|
|
* @private
|
|
*/
|
|
this._dragDrop = this._createDragDropHandlers();
|
|
}
|
|
|
|
/**
|
|
* The default options for the sheet.
|
|
* @type {DHSheetV2Configuration}
|
|
*/
|
|
static DEFAULT_OPTIONS = {
|
|
classes: ['daggerheart', 'sheet', 'dh-style'],
|
|
actions: {
|
|
createDoc: DHSheetV2.#createDoc,
|
|
editDoc: DHSheetV2.#editDoc,
|
|
deleteDoc: DHSheetV2.#deleteDoc
|
|
},
|
|
contextMenus: [],
|
|
dragDrop: [],
|
|
tagifyConfigs: []
|
|
};
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
/**@inheritdoc */
|
|
_attachPartListeners(partId, htmlElement, options) {
|
|
super._attachPartListeners(partId, htmlElement, options);
|
|
this._dragDrop.forEach(d => d.bind(htmlElement));
|
|
}
|
|
/**@inheritdoc */
|
|
async _onFirstRender(context, options) {
|
|
await super._onFirstRender(context, options);
|
|
if (!!this.options.contextMenus.length) this._createContextMenus();
|
|
}
|
|
|
|
/**@inheritdoc */
|
|
async _onRender(context, options) {
|
|
await super._onRender(context, options);
|
|
this._createTagifyElements(this.options.tagifyConfigs);
|
|
}
|
|
|
|
/* -------------------------------------------- */
|
|
/* Tags */
|
|
/* -------------------------------------------- */
|
|
|
|
/**
|
|
* Creates Tagify elements from configuration objects
|
|
* @param {TagifyConfig[]} tagConfigs - Array of Tagify configuration objects
|
|
* @throws {TypeError} If tagConfigs is not an array
|
|
* @throws {Error} If required properties are missing in config objects
|
|
* @param {TagifyConfig[]} tagConfigs
|
|
*/
|
|
_createTagifyElements(tagConfigs) {
|
|
if (!Array.isArray(tagConfigs)) throw new TypeError('tagConfigs must be an array');
|
|
|
|
tagConfigs.forEach(config => {
|
|
try {
|
|
const { selector, options, callback, tagifyOptions = {} } = config;
|
|
|
|
// Validate required fields
|
|
if (!selector || !options || !callback) {
|
|
throw new Error('Invalid TagifyConfig - missing required properties', config);
|
|
}
|
|
|
|
// Find target element
|
|
const element = this.element.querySelector(selector);
|
|
if (!element) {
|
|
throw new Error(`Element not found with selector: ${selector}`);
|
|
}
|
|
// Resolve dynamic options if function provided
|
|
const resolvedOptions = typeof options === 'function' ? options.call(this) : options;
|
|
|
|
// Initialize Tagify
|
|
tagifyElement(element, resolvedOptions, callback.bind(this), tagifyOptions);
|
|
} catch (error) {
|
|
console.error('Error initializing Tagify:', error);
|
|
}
|
|
});
|
|
}
|
|
|
|
/* -------------------------------------------- */
|
|
/* Drag and Drop */
|
|
/* -------------------------------------------- */
|
|
|
|
/**
|
|
* Creates drag-drop handlers from the configured options.
|
|
* @returns {foundry.applications.ux.DragDrop[]}
|
|
* @private
|
|
*/
|
|
_createDragDropHandlers() {
|
|
return this.options.dragDrop.map(d => {
|
|
d.callbacks = {
|
|
dragstart: this._onDragStart.bind(this),
|
|
drop: this._onDrop.bind(this)
|
|
};
|
|
return new foundry.applications.ux.DragDrop.implementation(d);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Handle dragStart event.
|
|
* @param {DragEvent} event
|
|
* @protected
|
|
*/
|
|
_onDragStart(event) {}
|
|
|
|
/**
|
|
* Handle drop event.
|
|
* @param {DragEvent} event
|
|
* @protected
|
|
*/
|
|
_onDrop(event) {}
|
|
|
|
/* -------------------------------------------- */
|
|
/* Context Menu */
|
|
/* -------------------------------------------- */
|
|
|
|
_createContextMenus() {
|
|
for (const config of this.options.contextMenus) {
|
|
const { handler, selector, options } = config;
|
|
this._createContextMenu(handler.bind(this), selector, options);
|
|
}
|
|
}
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
/**
|
|
* Get the set of ContextMenu options which should be used for journal entry pages in the sidebar.
|
|
* @returns {import('@client/applications/ux/context-menu.mjs').ContextMenuEntry[]}
|
|
* @protected
|
|
*/
|
|
_getEntryContextOptions() {
|
|
return [];
|
|
}
|
|
|
|
/* -------------------------------------------- */
|
|
/* Prepare Context */
|
|
/* -------------------------------------------- */
|
|
|
|
/**@inheritdoc*/
|
|
async _prepareContext(options) {
|
|
const context = await super._prepareContext(options);
|
|
context.config = CONFIG.DH;
|
|
context.source = this.document;
|
|
context.fields = this.document.schema.fields;
|
|
context.systemFields = this.document.system.schema.fields;
|
|
return context;
|
|
}
|
|
|
|
/* -------------------------------------------- */
|
|
/* Application Clicks Actions */
|
|
/* -------------------------------------------- */
|
|
|
|
/**
|
|
* Create an embedded document.
|
|
* @param {PointerEvent} event - The originating click event
|
|
* @param {HTMLElement} button - The capturing HTML element which defines the [data-action="removeAction"]
|
|
*/
|
|
static async #createDoc(event, button) {
|
|
const { documentClass, type } = button.dataset;
|
|
const parent = this.document;
|
|
|
|
const cls = getDocumentClass(documentClass);
|
|
return await cls.createDocuments(
|
|
[
|
|
{
|
|
name: cls.defaultName({ type, parent }),
|
|
type
|
|
}
|
|
],
|
|
{ parent, renderSheet: !event.shiftKey }
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Renders an embedded document.
|
|
* @param {PointerEvent} event - The originating click event
|
|
* @param {HTMLElement} button - The capturing HTML element which defines the [data-action="removeAction"]
|
|
*/
|
|
static #editDoc(_event, button) {
|
|
const { type, docId } = button.dataset;
|
|
this.document.getEmbeddedDocument(type, docId, { strict: true }).sheet.render({ force: true });
|
|
}
|
|
|
|
/**
|
|
* Delete an embedded document.
|
|
* @param {PointerEvent} _event - The originating click event
|
|
* @param {HTMLElement} button - The capturing HTML element which defines the [data-action="removeAction"]
|
|
*/
|
|
static async #deleteDoc(_event, button) {
|
|
const { type, docId } = button.dataset;
|
|
const document = this.document.getEmbeddedDocument(type, docId, { strict: true });
|
|
const typeName = game.i18n.localize(
|
|
document.type === 'base' ? `DOCUMENT.${type}` : `TYPES.${type}.${document.type}`
|
|
);
|
|
|
|
const confirmed = await foundry.applications.api.DialogV2.confirm({
|
|
window: {
|
|
title: game.i18n.format('DAGGERHEART.APPLICATIONS.DeleteConfirmation.title', {
|
|
type: typeName,
|
|
name: document.name
|
|
})
|
|
},
|
|
content: game.i18n.format('DAGGERHEART.APPLICATIONS.DeleteConfirmation.text', { name: document.name })
|
|
});
|
|
if (!confirmed) return;
|
|
|
|
await document.delete();
|
|
}
|
|
}
|
|
|
|
return DHSheetV2;
|
|
}
|