mirror of
https://github.com/Foundryborne/daggerheart.git
synced 2026-03-07 14:36:13 +01:00
162 lines
6.7 KiB
JavaScript
162 lines
6.7 KiB
JavaScript
import autocomplete from 'autocompleter';
|
|
|
|
export default class DhActiveEffectConfig extends foundry.applications.sheets.ActiveEffectConfig {
|
|
constructor(options) {
|
|
super(options);
|
|
|
|
const ignoredActorKeys = ['config', 'DhEnvironment'];
|
|
this.changeChoices = Object.keys(game.system.api.models.actors).reduce((acc, key) => {
|
|
if (!ignoredActorKeys.includes(key)) {
|
|
const model = game.system.api.models.actors[key];
|
|
const attributes = CONFIG.Token.documentClass.getTrackedAttributes(model);
|
|
// As per DHToken._getTrackedAttributesFromSchema, attributes.bar have a max version as well.
|
|
const maxAttributes = attributes.bar.map(x => [...x, 'max']);
|
|
attributes.value.push(...maxAttributes);
|
|
const group = game.i18n.localize(model.metadata.label);
|
|
const choices = CONFIG.Token.documentClass
|
|
.getTrackedAttributeChoices(attributes, model)
|
|
.map(x => ({ ...x, group: group }));
|
|
acc.push(...choices);
|
|
}
|
|
return acc;
|
|
}, []);
|
|
}
|
|
|
|
static DEFAULT_OPTIONS = {
|
|
classes: ['daggerheart', 'sheet', 'dh-style']
|
|
};
|
|
|
|
static PARTS = {
|
|
header: { template: 'systems/daggerheart/templates/sheets/activeEffect/header.hbs' },
|
|
tabs: { template: 'templates/generic/tab-navigation.hbs' },
|
|
details: { template: 'systems/daggerheart/templates/sheets/activeEffect/details.hbs', scrollable: [''] },
|
|
settings: { template: 'systems/daggerheart/templates/sheets/activeEffect/settings.hbs' },
|
|
changes: {
|
|
template: 'systems/daggerheart/templates/sheets/activeEffect/changes.hbs',
|
|
templates: ['systems/daggerheart/templates/sheets/activeEffect/change.hbs'],
|
|
scrollable: ['ol[data-changes]']
|
|
},
|
|
footer: { template: 'systems/daggerheart/templates/sheets/global/tabs/tab-form-footer.hbs' }
|
|
};
|
|
|
|
static TABS = {
|
|
sheet: {
|
|
tabs: [
|
|
{ id: 'details', icon: 'fa-solid fa-book' },
|
|
{ id: 'settings', icon: 'fa-solid fa-bars', label: 'DAGGERHEART.GENERAL.Tabs.settings' },
|
|
{ id: 'changes', icon: 'fa-solid fa-gears' }
|
|
],
|
|
initial: 'details',
|
|
labelPrefix: 'EFFECT.TABS'
|
|
}
|
|
};
|
|
|
|
_attachPartListeners(partId, htmlElement, options) {
|
|
super._attachPartListeners(partId, htmlElement, options);
|
|
const changeChoices = this.changeChoices;
|
|
|
|
htmlElement.querySelectorAll('.effect-change-input').forEach(element => {
|
|
autocomplete({
|
|
input: element,
|
|
fetch: function (text, update) {
|
|
if (!text) {
|
|
update(changeChoices);
|
|
} else {
|
|
text = text.toLowerCase();
|
|
var suggestions = changeChoices.filter(n => n.label.toLowerCase().includes(text));
|
|
update(suggestions);
|
|
}
|
|
},
|
|
render: function (item, search) {
|
|
const label = game.i18n.localize(item.label);
|
|
const matchIndex = label.toLowerCase().indexOf(search);
|
|
|
|
const beforeText = label.slice(0, matchIndex);
|
|
const matchText = label.slice(matchIndex, matchIndex + search.length);
|
|
const after = label.slice(matchIndex + search.length, label.length);
|
|
|
|
const element = document.createElement('li');
|
|
element.innerHTML = `${beforeText}${matchText ? `<strong>${matchText}</strong>` : ''}${after}`;
|
|
if (item.hint) {
|
|
element.dataset.tooltip = game.i18n.localize(item.hint);
|
|
}
|
|
|
|
return element;
|
|
},
|
|
renderGroup: function (label) {
|
|
const itemElement = document.createElement('div');
|
|
itemElement.textContent = game.i18n.localize(label);
|
|
return itemElement;
|
|
},
|
|
onSelect: function (item) {
|
|
element.value = `system.${item.value}`;
|
|
},
|
|
click: e => e.fetch(),
|
|
customize: function (_input, _inputRect, container) {
|
|
container.style.zIndex = foundry.applications.api.ApplicationV2._maxZ;
|
|
},
|
|
minLength: 0
|
|
});
|
|
});
|
|
}
|
|
|
|
async _prepareContext(options) {
|
|
const context = await super._prepareContext(options);
|
|
context.systemFields = context.document.system.schema.fields;
|
|
|
|
return context;
|
|
}
|
|
|
|
async _preparePartContext(partId, context) {
|
|
const partContext = await super._preparePartContext(partId, context);
|
|
switch (partId) {
|
|
case 'details':
|
|
const useGeneric = game.settings.get(
|
|
CONFIG.DH.id,
|
|
CONFIG.DH.SETTINGS.gameSettings.appearance
|
|
).showGenericStatusEffects;
|
|
if (!useGeneric) {
|
|
partContext.statuses = Object.values(CONFIG.DH.GENERAL.conditions()).map(status => ({
|
|
value: status.id,
|
|
label: game.i18n.localize(status.name)
|
|
}));
|
|
}
|
|
break;
|
|
case 'changes':
|
|
const fields = this.document.system.schema.fields.changes.element.fields;
|
|
partContext.changes = await Promise.all(
|
|
foundry.utils
|
|
.deepClone(context.source.changes)
|
|
.map((c, i) => this._prepareChangeContext(c, i, fields))
|
|
);
|
|
break;
|
|
}
|
|
|
|
return partContext;
|
|
}
|
|
|
|
_prepareChangeContext(change, index, fields) {
|
|
if (typeof change.value !== 'string') change.value = JSON.stringify(change.value);
|
|
const defaultPriority = game.system.api.documents.DhActiveEffect.CHANGE_TYPES[change.type]?.defaultPriority;
|
|
Object.assign(
|
|
change,
|
|
['key', 'type', 'value', 'priority'].reduce((paths, fieldName) => {
|
|
paths[`${fieldName}Path`] = `system.changes.${index}.${fieldName}`;
|
|
return paths;
|
|
}, {})
|
|
);
|
|
return (
|
|
game.system.api.documents.DhActiveEffect.CHANGE_TYPES[change.type].render?.(
|
|
change,
|
|
index,
|
|
defaultPriority
|
|
) ??
|
|
renderTemplate('systems/daggerheart/templates/sheets/activeEffect/change.hbs', {
|
|
change,
|
|
index,
|
|
defaultPriority,
|
|
fields
|
|
})
|
|
);
|
|
}
|
|
}
|