/** * Extend the basic ActorSheet with some very simple modifications * @extends {ActorSheet} */ export class frostgraveActorSheet extends ActorSheet { /** @override */ static get defaultOptions() { return mergeObject(super.defaultOptions, { classes: ["frostgrave", "sheet", "actor"], template: "systems/foundryvtt-frostgrave/templates/actor/actor-sheet.html", width: 650, height: 650, tabs: [{ navSelector: ".sheet-tabs", contentSelector: ".sheet-body", initial: "items", }, ], }); } /* -------------------------------------------- */ /** @override */ getData() { const data = super.getData(); let formData = { data: data.data, actor: this.actor, dtypes: ["String", "Number", "Boolean"] } //for (let attr of Object.values(data.data.attributes)) { // attr.isCheckbox = attr.dtype === "Boolean"; // } // Prepare items. if (this.actor.data.type == "character") { this._prepareCharacterItems(formData); } console.log("FORMDATA", formData); return formData; } /** @override */ activateListeners(html) { super.activateListeners(html); // Everything below here is only needed if the sheet is editable if (!this.options.editable) return; // Add Inventory Item html.find(".item-create").click(this._onItemCreate.bind(this)); // Edit Inventory Item html.find(".item-edit").click((ev) => { const card = $(ev.currentTarget).parents(".item-card"); const item = this.actor.items.get(card.data("item-id")); item.sheet.render(true); }); html.find(".weapon-attack a").click((ev) => { const card = $(ev.currentTarget).parents(".item-card"); const item = this.actor.items.get(card.data("item-id")); console.log("ACTOR: ", this.actor); this.actor.attackWeapon(item); }); // Delete Inventory Item html.find(".item-delete").click((ev) => { const card = $(ev.currentTarget).parents(".item-card"); this.actor.deleteOwnedItem(card.data("item-id")); }); // Rollable abilities. html.find(".rollable").click(this._onRoll.bind(this)); } /* -------------------------------------------- */ /** * Handle creating a new Owned Item for the actor using initial data defined in the HTML dataset * @param {Event} event The originating click event * @private */ _onItemCreate(event) { event.preventDefault(); const header = event.currentTarget; // Get the type of item to create. const type = header.dataset.type; // Grab any data associated with this control. const data = duplicate(header.dataset); // Initialize a default name. const name = `New ${type.capitalize()}`; // Prepare the item object. const itemData = { name: name, type: type, data: data, }; // Remove the type from the dataset since it's in the itemData.type prop. delete itemData.data["type"]; // Finally, create the item! return this.actor.createOwnedItem(itemData); } /** * Handle clickable rolls. * @param {Event} event The originating click event * @private */ _onRoll(event) { event.preventDefault(); const element = event.currentTarget; const dataset = element.dataset; if (dataset.roll) { let roll = new Roll(dataset.roll, this.actor.data.data ); let damage = parseInt(roll.evaluate( {async: false} ).total) + parseInt(dataset.bonus); let rollflavor = `${dataset.label} Roll: ` + roll.total; if (dataset.label == "Combat" || dataset.label == "Shooting") { let damageflavor = `
Damage: ` + damage; rollflavor = rollflavor + damageflavor; } roll.toMessage({ speaker: ChatMessage.getSpeaker({ actor: this.actor }), flavor: rollflavor, }); } if (dataset.spell) { let alignment = dataset.alignment; let empowerment = this.actor.data.data.empowerment; let alignmentmod; let selfdamage; let castresult; if (alignment == "Native") { alignmentmod = 0; } else if (alignment == "Aligned") { alignmentmod = 2; } else if (alignment == "Neutral") { alignmentmod = 4; } else { alignmentmod = 6; }; let difficulty = dataset.bcn - dataset.improved + alignmentmod; let roll = new Roll(`1d20+` + empowerment); //let roll = new Roll(`(1d20+` + empowerment + `)ms>=` + difficulty); roll.evaluate( { async: false}); let rollresult = difficulty - roll.total; if (rollresult >= 20) { selfdamage = 5; } else if (rollresult >= 10) { selfdamage = 2; } else if (rollresult >= 5) { selfdamage = 1; } else { selfdamage = 0; }; selfdamage = selfdamage + empowerment; if (rollresult <= 0) { castresult = 'SUCCESS'; } else { castresult = 'FAILURE'; }; let rollflavor = `Casting ${dataset.label} vs Difficulty ` + difficulty + `
` + castresult + `
BCN: ${dataset.bcn} | Alignment: +` + alignmentmod + ` | Improved: -${dataset.improved}
Empowerment: ` + empowerment + ` | Self Damage: ` + selfdamage; roll.toMessage({ speaker: ChatMessage.getSpeaker({ actor: this.actor }), flavor: rollflavor, }); } } /** * Organize and classify Items for Character sheets. * * @param {Object} actorData The actor to prepare. * * @return {undefined} */ _prepareCharacterItems(sheetData) { const actorData = sheetData.actor; // Initialize containers. const gear = []; const features = []; const spells = []; // Iterate through items, allocating to containers // let totalWeight = 0; for (let i of actorData.data.items) { let item = i.data; //i.img = i.img || DEFAULT_TOKEN; // Append to gear. if (i.type === "item") { gear.push(i); } // Append to features. else if (i.type === "feature") { features.push(i); } // Append to spells. else if (i.type === "spell") { spells.push(i); } } // Assign and return sheetData.gear = gear; sheetData.features = features; sheetData.spells = spells; } }