Adds skills and how to add them to the sheet

This commit is contained in:
mstein 2022-06-09 23:50:19 +02:00
parent 403109921c
commit 0f06d4020b
16 changed files with 478 additions and 162 deletions

View File

@ -136,5 +136,12 @@
"midgard5.spellBonus": "Zauberbonus", "midgard5.spellBonus": "Zauberbonus",
"midgard5.brawl": "Raufen", "midgard5.brawl": "Raufen",
"midgard5.poisonResistance": "Giftresistenz", "midgard5.poisonResistance": "Giftresistenz",
"midgard5.enduranceBonus": "Ausdauerbonus" "midgard5.enduranceBonus": "Ausdauerbonus",
"midgard5.new-skill": "Neue Fertigkeit",
"midgard5.special": "Spezial",
"midgard5.learned-skill": "Gelernte Fertigkeit",
"midgard5.language": "Sprache",
"midgard5.weapon-skill": "Waffenfertigkeit",
"midgard5.unlearned-skill": "Ungelernte Fertigkeit"
} }

View File

@ -7,7 +7,8 @@ const preloadTemplates = async (): Promise<Handlebars.TemplateDelegate<any>[]> =
// This would map to our local folder of /Assets/Templates/Actor/actor-sheet.hbs // This would map to our local folder of /Assets/Templates/Actor/actor-sheet.hbs
const templates: Array<string> = [ const templates: Array<string> = [
"sheets/character/base_values.hbs", "sheets/character/base_values.hbs",
"sheets/character/skills.hbs" "sheets/character/skills.hbs",
"sheets/character/gear.hbs"
] ]
return loadTemplates(templates.map(s => rootPath + s)) return loadTemplates(templates.map(s => rootPath + s))
} }

View File

@ -38,6 +38,10 @@ Hooks.once("init", async () => {
return actor.skillEw(skill).toString() return actor.skillEw(skill).toString()
}) })
Handlebars.registerHelper("skill", (skillId: string) => {
return (game as Game).items.get(skillId)
})
// Default Sheet für Items definieren und das Standardsheet deaktivieren // Default Sheet für Items definieren und das Standardsheet deaktivieren
Items.unregisterSheet("core", ItemSheet) Items.unregisterSheet("core", ItemSheet)
Items.registerSheet("midgard5", M5ItemSheet, { makeDefault: true }) Items.registerSheet("midgard5", M5ItemSheet, { makeDefault: true })

View File

@ -17,3 +17,10 @@ export interface M5Attribute {
value: number value: number
bonus: number bonus: number
} }
export interface M5RollData {
c: any,
i: any,
rolls: any,
res: any
}

View File

@ -1,21 +1,33 @@
import { M5Attribute, M5Skill, M5SkillLearned } from "../M5Base" import { M5Item } from "../items/M5Item"
import { M5Attribute, M5RollData, M5Skill, M5SkillLearned } from "../M5Base"
export interface M5CharacterCalculatedData { export interface M5CharacterCalculatedData {
level: number, level: number
attributes: {
st: { value: number, bonus: number }
gs: { value: number, bonus: number }
gw: { value: number, bonus: number }
ko: { value: number, bonus: number }
in: { value: number, bonus: number }
zt: { value: number, bonus: number }
au: { value: number, bonus: number }
pa: { value: number, bonus: number }
wk: { value: number, bonus: number }
}
stats: { stats: {
armor: number, armor: number
defense: number, defense: number
damageBonus: number, damageBonus: number
attackBonus: number, attackBonus: number
defenseBonus: number, defenseBonus: number
movementBonus: number, movementBonus: number
resistanceMind: number, resistanceMind: number
resistanceBody: number, resistanceBody: number
spellCasting: number, spellCasting: number
brawl: number, brawl: number
poisonResistance: number, poisonResistance: number
enduranceBonus: number enduranceBonus: number
}, }
skills: { skills: {
general: {} general: {}
} }
@ -49,10 +61,21 @@ export class M5Character extends Actor {
} }
prepareDerivedData() { prepareDerivedData() {
const context = (this as any).data; const context = (this as any).data
context.data.calc = { context.data.calc = {
level: 0, level: 0,
attributes: {
st: { value: 0, bonus: 0 },
gs: { value: 0, bonus: 0 },
gw: { value: 0, bonus: 0 },
ko: { value: 0, bonus: 0 },
in: { value: 0, bonus: 0 },
zt: { value: 0, bonus: 0 },
au: { value: 0, bonus: 0 },
pa: { value: 0, bonus: 0 },
wk: { value: 0, bonus: 0 }
},
stats: { stats: {
armor: 0, armor: 0,
defense: 0, defense: 0,
@ -68,41 +91,71 @@ export class M5Character extends Actor {
enduranceBonus: 0 enduranceBonus: 0
}, },
skills: { skills: {
general: {} general: {},
combat: {},
language: {},
custom: {}
} }
} as M5CharacterCalculatedData } as M5CharacterCalculatedData
const data = context.data const data = context.data
const st = M5Character.attributeMinMax(data.attributes.st)
const gs = M5Character.attributeMinMax(data.attributes.gs)
const gw = M5Character.attributeMinMax(data.attributes.gw)
const ko = M5Character.attributeMinMax(data.attributes.ko)
const calc = context.data.calc as M5CharacterCalculatedData const calc = context.data.calc as M5CharacterCalculatedData
calc.level = M5Character.levelFromExp(data.es) calc.level = M5Character.levelFromExp(data.es)
calc.attributes.st.value = M5Character.attributeMinMax(data.attributes.st) // TODO item effects
calc.attributes.gs.value = M5Character.attributeMinMax(data.attributes.gs)
calc.attributes.gw.value = M5Character.attributeMinMax(data.attributes.gw)
calc.attributes.ko.value = M5Character.attributeMinMax(data.attributes.ko)
calc.attributes.in.value = M5Character.attributeMinMax(data.attributes.in)
calc.attributes.zt.value = M5Character.attributeMinMax(data.attributes.zt)
calc.attributes.au.value = M5Character.attributeMinMax(data.attributes.au)
calc.attributes.pa.value = M5Character.attributeMinMax(data.attributes.pa)
calc.attributes.wk.value = M5Character.attributeMinMax(data.attributes.wk)
calc.attributes.st.bonus = M5Character.attributeBonus(data.attributes.st)
calc.attributes.gs.bonus = M5Character.attributeBonus(data.attributes.gs)
calc.attributes.gw.bonus = M5Character.attributeBonus(data.attributes.gw)
calc.attributes.ko.bonus = M5Character.attributeBonus(data.attributes.ko)
calc.attributes.in.bonus = M5Character.attributeBonus(data.attributes.in)
calc.attributes.zt.bonus = M5Character.attributeBonus(data.attributes.zt)
calc.attributes.au.bonus = M5Character.attributeBonus(data.attributes.au)
calc.attributes.pa.bonus = M5Character.attributeBonus(data.attributes.pa)
calc.attributes.wk.bonus = M5Character.attributeBonus(data.attributes.wk)
calc.stats.armor = 0 calc.stats.armor = 0
calc.stats.defense = M5Character.defenseFromLevel(calc.level) calc.stats.defense = M5Character.defenseFromLevel(calc.level)
calc.stats.damageBonus = Math.floor(st/20) + Math.floor(gs/30) - 3 calc.stats.damageBonus = Math.floor(calc.attributes.st.value/20) + Math.floor(calc.attributes.gs.value/30) - 3
calc.stats.attackBonus = M5Character.attributeBonus(data.attributes.gs) calc.stats.attackBonus = calc.attributes.gs.bonus
calc.stats.defenseBonus = M5Character.attributeBonus(data.attributes.gw) calc.stats.defenseBonus = calc.attributes.gw.bonus
calc.stats.movementBonus = 0 calc.stats.movementBonus = 0
calc.stats.resistanceMind = calc.stats.defense calc.stats.resistanceMind = calc.stats.defense
calc.stats.resistanceBody = calc.stats.defense + 1 calc.stats.resistanceBody = calc.stats.defense + 1
calc.stats.spellCasting = (data.info.magicUsing ? M5Character.spellCastingFromLevel(calc.level) : 3) + M5Character.attributeBonus(data.attributes.zt) calc.stats.spellCasting = (data.info.magicUsing ? M5Character.spellCastingFromLevel(calc.level) : 3) + calc.attributes.zt.bonus
calc.stats.brawl = Math.floor((st + gw) / 20) calc.stats.brawl = Math.floor((calc.attributes.st.value + calc.attributes.gw.value) / 20)
calc.stats.poisonResistance = 30 + Math.floor(ko / 2) calc.stats.poisonResistance = 30 + Math.floor(calc.attributes.ko.value / 2)
calc.stats.enduranceBonus = Math.floor(ko/10) + Math.floor(st/20) calc.stats.enduranceBonus = Math.floor(calc.attributes.ko.value/10) + Math.floor(calc.attributes.st.value/20)
//console.log("prepareDerivedData", context)
context.items?.filter(item => item.data.type === "skill").forEach(item => { context.items?.filter(item => item.data.type === "skill").forEach(item => {
calc.skills.general[item.data.name] = { item.prepareDerivedData()
const skillMap = calc.skills[item.data.data.type]
skillMap[item.data._id] = {
label: item.data.name,
fw: item.data.data.fw, fw: item.data.data.fw,
attribute: item.data.data.attribute, attribute: item.data.data.attribute,
id: item.data._id calc: item.data.data.calc
} }
}) })
} }
override getRollData(): any {
return {
c: (this as any).data.data,
i: null,
rolls: {},
res: {}
} as M5RollData
}
static readonly levelThreshold: Array<number> = [0, 100, 250, 500, 750, 1000, 1250, 1500, 1750, 2000, 2500, 3000, 3500, 4000, 4500, 5000, 6000, 7000, 8000, 9000, 10000, 12500, 15000, 17500, 20000, 22500, 25000, 30000, 35000, 40000, 45000, 50000, 55000, 60000, 65000, 70000, 75000, 80000, 85000, 90000, 95000, 100000, 105000, 110000, 115000, 120000, 125000, 130000, 135000, 140000, 145000, 150000, 155000, 160000, 165000, 170000, 175000, 180000, 185000, 190000, 195000, 200000, 205000, 210000, 215000, 220000, 225000, 230000, 235000, 240000, 245000, 250000, 255000, 260000, 265000, 270000, 275000, 280000] static readonly levelThreshold: Array<number> = [0, 100, 250, 500, 750, 1000, 1250, 1500, 1750, 2000, 2500, 3000, 3500, 4000, 4500, 5000, 6000, 7000, 8000, 9000, 10000, 12500, 15000, 17500, 20000, 22500, 25000, 30000, 35000, 40000, 45000, 50000, 55000, 60000, 65000, 70000, 75000, 80000, 85000, 90000, 95000, 100000, 105000, 110000, 115000, 120000, 125000, 130000, 135000, 140000, 145000, 150000, 155000, 160000, 165000, 170000, 175000, 180000, 185000, 190000, 195000, 200000, 205000, 210000, 215000, 220000, 225000, 230000, 235000, 240000, 245000, 250000, 255000, 260000, 265000, 270000, 275000, 280000]
static levelFromExp(exp: number): number { static levelFromExp(exp: number): number {
const ret = M5Character.levelThreshold.findIndex(val => val > exp) const ret = M5Character.levelThreshold.findIndex(val => val > exp)
@ -155,4 +208,16 @@ export class M5Character extends Actor {
return context.data.attributes[name] return context.data.attributes[name]
} }
createSkill(skillName: string): Promise<M5Item> {
const itemData = {
name: skillName,
type: "skill"
};
return (this as any).createEmbeddedDocuments("Item", [itemData]).then(docs => {
const item = docs[0]
return item as M5Item
})
}
} }

View File

@ -1,31 +1,45 @@
import { M5Character } from "../actors/M5Character" import { M5Character } from "../actors/M5Character"
import { M5Skill } from "../M5Base" import { M5RollData, M5Skill } from "../M5Base"
export class M5Item extends Item { export class M5Item extends Item {
static readonly SKILL = "skill" static readonly SKILL = "skill"
getRollData() { prepareDerivedData() {
if (!this.actor) const actor = (this.actor as any)
return null const character = actor as M5Character
const context = (this as any).data
const calc = context.data.calc
if (context.type === "skill") {
calc.bonus = 0
if (context.data.attribute && context.data.attribute !== "") {
//console.log(context.name, context.data)
const attribute = character.attribute(context.data.attribute)
calc.bonus += M5Character.attributeBonus(attribute)
}
if (context._id === actor.data.data.skills.preferredCombatSkill) {
calc.bonus += 2
}
calc.ew = context.data.fw + calc.bonus
} else if (context.type === "weapon") {
}
}
getRollData() {
const actor = this.actor as any const actor = this.actor as any
const context = (this as any).data const context = (this as any).data
let ret = null let ret: M5RollData = actor?.getRollData() ?? {
if (context.type === "skill") { c: null,
const character = actor as M5Character i: null,
const actorData = actor.data.data rolls: {},
const itemData = context.data as M5Skill res: {}
ret = {
fw: itemData.fw,
bonus: M5Character.attributeBonus(character.attribute(itemData.attribute))
}
} else {
ret = actor.getRollData()
ret.item = foundry.utils.deepClone(context.data)
} }
ret.i = context.data
return ret return ret
} }
@ -38,27 +52,32 @@ export class M5Item extends Item {
const label = `[${item.type}] ${item.name}` const label = `[${item.type}] ${item.name}`
// If there's no roll data, send a chat message. // If there's no roll data, send a chat message.
if (!item.data.formula) { const formulaNames = item.data.rolls?.formulas ? Object.keys(item.data.rolls.formulas) : []
if (formulaNames.length > 0) {
const rollData = this.getRollData()
let ret = []
formulaNames.forEach(formulaName => {
const formula = item.data.rolls.formulas[formulaName]
const roll = new Roll(formula.formula, rollData)
roll.toMessage({
speaker: speaker,
rollMode: rollMode,
flavor: label,
})
ret.push(roll)
})
return ret
} else {
ChatMessage.create({ ChatMessage.create({
speaker: speaker, speaker: speaker,
rollMode: rollMode, rollMode: rollMode,
flavor: label, flavor: label,
content: item.data.description ?? '' content: item.data.description ?? ''
}) })
} else { // Otherwise, create a roll and send a chat message from it. return null
// Retrieve roll data.
const rollData = this.getRollData()
// Invoke the roll and submit it to chat.
const roll = new Roll(item.data.formula, rollData)
// If you need to store the value first, uncomment the next line.
// let result = await roll.roll({async: true})
roll.toMessage({
speaker: speaker,
rollMode: rollMode,
flavor: label,
})
return roll
} }
} }

View File

@ -9,7 +9,7 @@ export default class M5CharacterSheet extends ActorSheet {
return mergeObject(super.defaultOptions, { return mergeObject(super.defaultOptions, {
template: "systems/midgard5/templates/sheets/character/main.hbs", template: "systems/midgard5/templates/sheets/character/main.hbs",
width: 600, width: 600,
height: 400, height: 800,
classes: ["midgard5", "sheet", "character"], classes: ["midgard5", "sheet", "character"],
tabs: [{ navSelector: ".sheet-navigation", contentSelector: ".sheet-content", initial: "base_values" }] tabs: [{ navSelector: ".sheet-navigation", contentSelector: ".sheet-content", initial: "base_values" }]
}) })
@ -46,6 +46,20 @@ export default class M5CharacterSheet extends ActorSheet {
item.sheet.render(true) item.sheet.render(true)
}) })
html.find(".change-special-combat-skill").on("click", async (event) => {
const row = event.target.parentElement.parentElement
let skillId = row.dataset["skill"]
const actor = this.actor as any
actor.update({
data: {
skills: {
preferredCombatSkill: skillId
}
}
})
})
html.find(".roll-learned-button").on("click", async (event) => { html.find(".roll-learned-button").on("click", async (event) => {
const row = event.target.parentElement.parentElement const row = event.target.parentElement.parentElement
let skillId = row.dataset["skill"] let skillId = row.dataset["skill"]
@ -58,6 +72,24 @@ export default class M5CharacterSheet extends ActorSheet {
await item.roll() await item.roll()
}) })
html.find(".create-skill-button").on("click", async (event) => {
const button = event.target
const skillType = button.dataset["skilltype"]
const character = this.actor as M5Character
character.createSkill((game as Game).i18n.localize("midgard5.new-skill")).then(skill => {
const item = skill as any
item.update({
data: {
fw: 5,
attribute: "",
skill: "",
type: skillType
}
})
})
})
html.find(".roll-general-button").on("click", async (event) => { html.find(".roll-general-button").on("click", async (event) => {
const row = event.target.parentElement.parentElement const row = event.target.parentElement.parentElement
let skillName = row.dataset["skill"] let skillName = row.dataset["skill"]
@ -93,30 +125,89 @@ export default class M5CharacterSheet extends ActorSheet {
const data = this.actor.data.data const data = this.actor.data.data
const unlearnedSkill = data.skills.general[skillName] as M5SkillUnlearned const unlearnedSkill = data.skills.general[skillName] as M5SkillUnlearned
const itemData = { const character = this.actor as M5Character
name: (game as Game).i18n.localize("midgard5." + skillName), character.createSkill((game as Game).i18n.localize("midgard5." + skillName)).then(skill => {
type: "skill", const item = skill as any
item.update({
data: { data: {
fw: unlearnedSkill.initial, fw: unlearnedSkill.initial,
attribute: unlearnedSkill.attribute, attribute: unlearnedSkill.attribute,
skill: skillName, skill: skillName,
type: "general" type: "general"
} as M5SkillLearned
} }
})
this.actor.createEmbeddedDocuments("Item", [itemData]).then(docs => {
const item = docs[0]
//console.log("createEmbeddedDocuments", item.data)
}) })
}) })
// Drag & Drop // Drag & Drop
// const dragDropSkill = new DragDrop({ const dragDrop = new DragDrop({
// dragSelector: ".items-list .item", dragSelector: ".items-list .item",
// dropSelector: ".sheet-body", dropSelector: ".sheet-body",
// permissions: { dragstart: this._canDragStart.bind(this), drop: this._canDragDrop.bind(this) }, permissions: { dragstart: this._canDragStart.bind(this), drop: this._canDragDrop.bind(this) },
// callbacks: { dragstart: this._onTransferItemDragStart.bind(this), drop: this._onTransferItemDrop.bind(this) }, callbacks: { dragstart: this._onTransferItemDragStart.bind(this), drop: this._onTransferItemDrop.bind(this) },
// }) })
// dragDrop.bind(html[0]) dragDrop.bind(html[0])
}
_canDragStart(selector) {
return this.options.editable && this.actor.isOwner;
}
_canDragDrop(selector) {
return true;
}
_onTransferItemDragStart(event) {
const li = event.currentTarget
$(event.currentTarget).attr("data-item-actorid", this.actor.id)
const item = this.actor.items.get(li.dataset.itemId)
// limit transfer on personal weapons/armour/gear
if (["weapon"].includes(item.data.type)) {
const dragData = {
type: "Transfer",
actorId: this.actor.id,
data: item.data,
tokenId: null
}
if (this.actor.isToken)
dragData.tokenId = this.actor.token.id
event.dataTransfer.setData("text/plain", JSON.stringify(dragData))
} else {
return false
} }
} }
async _onTransferItemDrop(event) {
// Try to extract the data
let data = null
try {
data = JSON.parse(event.dataTransfer.getData("text/plain"))
if (data.type !== "Transfer")
return false
} catch (err) {
return false
}
if (!data.data)
return false
if (data.actorId === this.actor.id)
return false
try {
this.actor.createEmbeddedDocuments("Item", [duplicate(data.data)]) // Create a new Item
const actor = (game as any).actors.get(data.actorId)
await actor.items.get(data.data._id)?.delete() // Delete originating item from other actor
} catch (e) {
console.error("Error transfering item between actors", e)
return false
}
return true
}
}

View File

@ -1,10 +1,11 @@
import { M5Item } from "../items/M5Item"
export class M5ItemSheet extends ItemSheet { export class M5ItemSheet extends ItemSheet {
static get defaultOptions() { static get defaultOptions() {
return mergeObject(super.defaultOptions, { return mergeObject(super.defaultOptions, {
width: 530, width: 420,
height: 340, height: 240,
classes: ["midgard5", "sheet", "item"] classes: ["midgard5", "sheet", "item"]
}) })
} }
@ -16,7 +17,9 @@ export class M5ItemSheet extends ItemSheet {
} }
override getData(options?: Partial<ItemSheet.Options>): ItemSheet.Data<ItemSheet.Options> | Promise<ItemSheet.Data<ItemSheet.Options>> { override getData(options?: Partial<ItemSheet.Options>): ItemSheet.Data<ItemSheet.Options> | Promise<ItemSheet.Data<ItemSheet.Options>> {
const item = this.item as M5Item
return Promise.resolve(super.getData()).then(value => { return Promise.resolve(super.getData()).then(value => {
item.prepareDerivedData()
const context = value as any const context = value as any
// Use a safe clone of the item data for further operations. // Use a safe clone of the item data for further operations.

View File

@ -42,16 +42,24 @@
} }
td { td {
padding: 0 0.5rem 0 0.5rem;
&.center { &.center {
text-align: center; text-align: center;
} }
&.padding {
padding: 0 1rem 0 1rem;
}
} }
input.skill { input.skill {
width: 5rem; width: 5rem;
} }
.new-skill {
font-style: italic;
background: rgba(0, 0, 0, 0.3);
color:rgba(255, 255, 255);
button {
background: rgba(255, 255, 255, 0.5);
}
}
} }

View File

@ -40,47 +40,21 @@
}, },
"attributes": { "attributes": {
"attributes": { "attributes": {
"st": { "st": { "value": 50, "bonus": 0 },
"value": 50, "gs": { "value": 50, "bonus": 0 },
"bonus": 0 "gw": { "value": 50, "bonus": 0 },
}, "ko": { "value": 50, "bonus": 0 },
"gs": { "in": { "value": 50, "bonus": 0 },
"value": 50, "zt": { "value": 50, "bonus": 0 },
"bonus": 0 "au": { "value": 50, "bonus": 0 },
}, "pa": { "value": 50, "bonus": 0 },
"gw": { "wk": { "value": 50, "bonus": 0 }
"value": 50,
"bonus": 0
},
"ko": {
"value": 50,
"bonus": 0
},
"in": {
"value": 50,
"bonus": 0
},
"zt": {
"value": 50,
"bonus": 0
},
"au": {
"value": 50,
"bonus": 0
},
"pa": {
"value": 50,
"bonus": 0
},
"wk": {
"value": 50,
"bonus": 0
}
} }
}, },
"skills": { "skills": {
"skills": { "skills": {
"preferredCombatSkill": "", "preferredCombatSkill": "",
"innate": {},
"general": { "general": {
"akrobatik": { "fw": 6, "attribute": "gw", "initial": 8 }, "akrobatik": { "fw": 6, "attribute": "gw", "initial": 8 },
"alchimie": { "fw": 0, "attribute": "in", "initial": 8 }, "alchimie": { "fw": 0, "attribute": "in", "initial": 8 },
@ -139,7 +113,8 @@
} }
}, },
"character": { "character": {
"templates": ["characterBars", "attributes", "characterDescription", "characterHeader", "skills"] "templates": ["characterBars", "attributes", "characterDescription", "characterHeader", "skills"],
"calc": {}
}, },
"npc": { "npc": {
"templates": ["characterBars", "attributes", "characterDescription"] "templates": ["characterBars", "attributes", "characterDescription"]
@ -175,6 +150,16 @@
"pa": { "short": "midgard5.actor-pa", "long": "midgard5.actor-pa-long" }, "pa": { "short": "midgard5.actor-pa", "long": "midgard5.actor-pa-long" },
"wk": { "short": "midgard5.actor-wk", "long": "midgard5.actor-wk-long" } "wk": { "short": "midgard5.actor-wk", "long": "midgard5.actor-wk-long" }
} }
},
"rollable": {
"rolls": {
"formulas": {},
"output": ""
}
},
"equippable": {
"equippable": false,
"equipped": true
} }
}, },
"skill": { "skill": {
@ -183,27 +168,52 @@
"attribute": "st", "attribute": "st",
"skill": "", "skill": "",
"type": "general", "type": "general",
"formula": "1d20 + @fw + @bonus" "rolls": {
"formulas": {
"1": {
"formula": "1d20 + @i.fw + @i.calc.bonus",
"type": "ew"
}
},
"output": ""
},
"calc": {}
}, },
"item": { "item": {
"templates": ["itemDescription"], "templates": ["itemDescription", "stats", "equippable"],
"quantity": 1, "quantity": 1,
"value": 0, "value": 0,
"magic": false, "magic": false,
"onbody": false, "onbody": false,
"attributes": {}, "attributes": {},
"groups": {} "groups": {},
"rolls": {
"formulas": {},
"output": ""
}
}, },
"weapon": { "weapon": {
"templates": ["itemDescription", "stats"], "templates": ["itemDescription", "stats", "equippable"],
"skill": "" "defensive": false,
"skill": "",
"rolls": {
"formulas": {},
"output": ""
}
}, },
"armor": { "armor": {
"templates": ["itemDescription", "stats"], "templates": ["itemDescription", "stats", "equippable"],
"skill": "" "rolls": {
"formulas": {},
"output": ""
}
}, },
"spell": { "spell": {
"templates": ["itemDescription"] "templates": ["itemDescription"],
"rolls": {
"formulas": {},
"output": ""
}
} }
} }
} }

View File

@ -16,15 +16,15 @@ class Logger {
console.log(time.str + moduleName.str + text.str, ...time.params.concat(moduleName.params, text.params)); console.log(time.str + moduleName.str + text.str, ...time.params.concat(moduleName.params, text.params));
} }
static Err(str: string): void { static err(str: string): void {
Logger.log(str, Color("orange")); Logger.log(str, Color("orange"));
} }
static Warn(str: string): void { static warn(str: string): void {
Logger.log(str, Color("yellow")); Logger.log(str, Color("yellow"));
} }
static Ok(str: string): void { static ok(str: string): void {
Logger.log(str, Color("green")); Logger.log(str, Color("green"));
} }
} }

View File

View File

@ -22,5 +22,9 @@
{{> "systems/midgard5/templates/sheets/character/skills.hbs"}} {{> "systems/midgard5/templates/sheets/character/skills.hbs"}}
</div> </div>
<div class="tab base_values flexcol" data-group="primary" data-tab="gear">
{{> "systems/midgard5/templates/sheets/character/gear.hbs"}}
</div>
</section> </section>
</form> </form>

View File

@ -1,8 +1,7 @@
<h3>Gelernte Fertigkeiten</h3>
<table> <table>
<thead> <thead>
<tr> <tr>
<th>{{localize "midgard5.skill"}}</th> <th>{{localize "midgard5.learned-skill"}}</th>
<th>{{localize "midgard5.fw"}}</th> <th>{{localize "midgard5.fw"}}</th>
<th>{{localize "midgard5.bonus"}}</th> <th>{{localize "midgard5.bonus"}}</th>
<th>{{localize "midgard5.ew"}}</th> <th>{{localize "midgard5.ew"}}</th>
@ -10,24 +9,102 @@
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
{{#each data.calc.skills.general as |skill label|}} {{#each data.calc.skills.general as |skill skillId|}}
<tr data-skill="{{skill.id}}"> <tr data-skill="{{skillId}}">
<td class="padding edit-skill">{{label}}</td> <td class="padding edit-skill">{{skill.label}}</td>
<td class="center">{{skill.fw}}</td> <td class="center">{{skill.fw}}</td>
<td class="center">{{skillBonus ../actor._id skill}}</td> <td class="center">{{skill.calc.bonus}}</td>
<td class="center">{{skillEw ../actor._id skill}}</td> <td class="center">{{skill.calc.ew}}</td>
<td><button class="roll-learned-button">{{localize "midgard5.roll"}}</button></td> <td><button class="roll-learned-button">{{localize "midgard5.roll"}}</button></td>
</tr> </tr>
{{/each}} {{/each}}
<tr class="new-skill">
<td class="padding">{{localize "midgard5.new-skill"}}</td>
<td></td>
<td></td>
<td></td>
<td><button data-skilltype="general" class="create-skill-button">{{localize "midgard5.learn"}}</button></td>
</tr>
</tbody>
</table>
<table>
<thead>
<tr>
<th>{{localize "midgard5.language"}}</th>
<th>{{localize "midgard5.fw"}}</th>
<th>{{localize "midgard5.bonus"}}</th>
<th>{{localize "midgard5.ew"}}</th>
<th></th>
</tr>
</thead>
<tbody>
{{#each data.calc.skills.language as |skill skillId|}}
<tr data-skill="{{skillId}}">
<td class="padding edit-skill">{{skill.label}}</td>
<td class="center">{{skill.fw}}</td>
<td class="center">{{skill.calc.bonus}}</td>
<td class="center">{{skill.calc.ew}}</td>
<td><button class="roll-learned-button">{{localize "midgard5.roll"}}</button></td>
</tr>
{{/each}}
<tr class="new-skill">
<td class="padding">{{localize "midgard5.new-skill"}}</td>
<td></td>
<td></td>
<td></td>
<td><button data-skilltype="language" class="create-skill-button">{{localize "midgard5.learn"}}</button></td>
</tr>
</tbody>
</table>
<table>
<thead>
<tr>
<th>{{localize "midgard5.weapon-skill"}}</th>
<th>{{localize "midgard5.special"}}</th>
<th>{{localize "midgard5.fw"}}</th>
<th>{{localize "midgard5.bonus"}}</th>
<th>{{localize "midgard5.ew"}}</th>
<th></th>
</tr>
</thead>
<tbody>
{{#each data.calc.skills.combat as |skill skillId|}}
<tr data-skill="{{skillId}}">
<td class="padding edit-skill">{{skill.label}}</td>
<td class="center">
{{#if (eq skillId ../data.skills.preferredCombatSkill)}}
<input type="checkbox" checked />
{{else}}
<input type="checkbox" class="change-special-combat-skill" />
{{/if}}
</td>
<td class="center">{{skill.fw}}</td>
<td class="center">{{skill.calc.bonus}}</td>
<td class="center">{{skill.calc.ew}}</td>
<td><button class="roll-learned-button">{{localize "midgard5.roll"}}</button></td>
</tr>
{{/each}}
<tr class="new-skill">
<td class="padding">{{localize "midgard5.new-skill"}}</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td><button data-skilltype="combat" class="create-skill-button">{{localize "midgard5.learn"}}</button></td>
</tr>
</tbody> </tbody>
</table> </table>
<h3>Ungelernte Fertigkeiten</h3>
<table> <table>
<thead> <thead>
<tr> <tr>
<th></th> <th></th>
<th>{{localize "midgard5.skill"}}</th> <th>{{localize "midgard5.unlearned-skill"}}</th>
<th>{{localize "midgard5.fw"}}</th> <th>{{localize "midgard5.fw"}}</th>
<th>{{localize "midgard5.bonus"}}</th> <th>{{localize "midgard5.bonus"}}</th>
<th>{{localize "midgard5.ew"}}</th> <th>{{localize "midgard5.ew"}}</th>

View File

@ -4,11 +4,17 @@
<h1><input name="name" type="text" value="{{item.name}}" placeholder="Name" /></h1> <h1><input name="name" type="text" value="{{item.name}}" placeholder="Name" /></h1>
</header> </header>
<div class="sheet-content"> <div class="sheet-content">
<div class="flexrow"> <table>
<span>{{localize "midgard5.skill-value"}}</span> <tbody>
<input name="data.fw" type="text" value="{{data.fw}}" data-dtype="Number" /> <tr>
<td>{{localize "midgard5.skill-value"}}</td>
<td><input name="data.fw" type="text" value="{{data.fw}}" data-dtype="Number" /></td>
</tr>
<span>{{localize "midgard5.attribute"}}</span> {{#unless (eq data.type "combat")}}
<tr>
<td>{{localize "midgard5.attribute"}}</td>
<td>
<select class="select-attribute" name="data.attribute" data-type="String"> <select class="select-attribute" name="data.attribute" data-type="String">
{{#select data.attribute}} {{#select data.attribute}}
{{#each data.attributes as |attribute key|}} {{#each data.attributes as |attribute key|}}
@ -16,7 +22,11 @@
{{/each}} {{/each}}
{{/select}} {{/select}}
</select> </select>
</div> </td>
</tr>
{{/unless}}
</tbody>
</table>
{{editor content=data.description target="data.description" button=true owner=owner editable=editable}} {{editor content=data.description target="data.description" button=true owner=owner editable=editable}}
</div> </div>

View File

@ -0,0 +1,10 @@
<form class="{{cssClass}}" autocomplete="off">
<header class="sheet-header">
<img class="item-img" src="{{item.img}}" data-edit="img" title="{{item.name}}" />
<h1><input name="name" type="text" value="{{item.name}}" placeholder="Name" /></h1>
</header>
<div class="sheet-content">
{{editor content=data.description target="data.description" button=true owner=owner editable=editable}}
</div>
</form>