diff --git a/RolemasterUnified_Official/rolemasterunified.css b/RolemasterUnified_Official/rolemasterunified.css index 0530b9e12a6..7bbdc3adcbc 100644 --- a/RolemasterUnified_Official/rolemasterunified.css +++ b/RolemasterUnified_Official/rolemasterunified.css @@ -710,6 +710,7 @@ div.repcontrol { .cm_info { border: 3px solid rgb(20,21,22); border-radius: 255px 15px 225px 15px/15px 225px 15px 255px; + height: 80%; } diff --git a/RolemasterUnified_Official/rolemasterunified.html b/RolemasterUnified_Official/rolemasterunified.html index b980148c8d1..725ead55c90 100644 --- a/RolemasterUnified_Official/rolemasterunified.html +++ b/RolemasterUnified_Official/rolemasterunified.html @@ -204,8 +204,6 @@ - -
@@ -2115,6 +2113,9 @@

+ + +
@@ -5096,7 +5097,14 @@

Recalculate

+
+

Rerun the initial charactermancer. Warning this may add all sorts of weird results to your character. +This is not for usual use.

+ + + +

This is the magic button to apply attack results. Clicking should do little.


@@ -5106,7 +5114,7 @@

Recalculate


-Revision 7c647d878cc463743cc3ecad22079f37bf249bd6 +Revision 312226f26becda9be73d84535e75bcb5c7156fe7
@@ -8230,81 +8238,108 @@

Recalculate

}); - // So we need to get the profession & Costs - const data = getCharmancerData(); + let costs = {}; - const costs = getCostsWithWeapons(data.knacks.values); - dpspent = 0; + addPendingFunction("Level Up fetch costs", () => { + const toget = []; + categories.forEach(cat => { + toget.push(cat.aname + '_cost'); + }); + toget.push('magicalritual_cost', 'spellownbase_cost', 'spellopen_cost', + 'spallclosed_cost', 'spellrestricted_cost', 'spellarcane_cost'); + toget.push('meleeweapons_cost', 'rangedweapons_cost', 'unarmed_cost', + 'shield_cost'); + console.log("toget for level up", toget); + getAttrsPending(toget, (results) => { + // Strip the costs of each key + for (const [key, cost] of Object.entries(results)) { + costs[key.replace(/_cost/, "")] = cost; + } + }); + }); - let catupdates = {} - for (const cat in costs) { - catupdates['cmcategory-' + cat + '-cost'] = costs[cat]; - } - catupdates['cmcategory-combattraining-cost'] = '-'; - catupdates['cmcategory-spellcasting-cost'] = '-'; - setCharmancerText(catupdates); + // Global scope; as the other is a pending funtion. + let savedspelllists = []; + let savedtalents = []; + let savedstatgains = []; - // Update each of the cost fields and the selectors. - RMUSkills.forEachSkill((skill, cat, pskill) => { - if (skill.spellgroup) { - return; - } - const skillname = pskill ? pskill.aname : skill.aname; - const costarray = generateCostArray(costs[cat.aname] ?? costs[skillname]); - if (costarray.length == 0) { - rmuerror("Could not generate costs", pskill, pskill.aname, skill.aname); - return; - } - cmPopulateListOptions('cmskill' + skill.aname + 'ranks', costarray); - }); + addPendingFunction("Level Up Start body", () => { + // So we need to get the profession & Costs - // Global scope; as the other is a pending funtion. - let savedspelllists = []; - let savedtalents = []; - let savedstatgains = []; - // On load I should check the list of repeating sections and add - // them - if (data?.levelup?.repeating) { - const profession = data.profession.values.profession; - const costs = getCostsFromData(profession); - for (const index in data.levelup.repeating) { - // Need to extract the skillname so I can call addRepeating again - const prefix = data.levelup.repeating[index]; - // See if it's spell list. - if (prefix.includes("spelllist")) { - savedspelllists.push(prefix); - continue; - } - if (prefix.endsWith("_talent")) { - savedtalents.push(prefix); - continue; - } - if (prefix.endsWith("_statgain")) { - savedstatgains.push(prefix); - continue; - } - const match = prefix.match(/cmskill_(.*)_specializations/) - if (!match || match.length < 2) { - rmuerror("Coulnd't match regex against ", prefix, index); - continue; + + const data = getCharmancerData(); + +// const costs = getCostsWithWeapons(data.knacks.values); + dpspent = 0; + + let catupdates = {} + for (const cat in costs) { + catupdates['cmcategory-' + cat + '-cost'] = costs[cat]; } - const origname = match[0]; - const skillname = match[1]; - const skill = RMUSkills.getSkillByName(skillname) - const costarray = generateCostArray(costs[skill.category]); - addRepeatingSection(`cmskill${skillname}`, 'specialization', origname, - (sectionname) => { + catupdates['cmcategory-combattraining-cost'] = '-'; + catupdates['cmcategory-spellcasting-cost'] = '-'; + setCharmancerText(catupdates); + + // Update each of the cost fields and the selectors. + RMUSkills.forEachSkill((skill, cat, pskill) => { + if (skill.spellgroup) { + return; + } + const skillname = pskill ? pskill.aname : skill.aname; + const costarray = generateCostArray(costs[cat.aname] ?? costs[skillname]); + if (costarray.length == 0) { + rmuerror("Could not generate costs", pskill, pskill.aname, skill.aname); + return; + } + cmPopulateListOptions('cmskill' + skill.aname + 'ranks', costarray); + }); + + + // On load I should check the list of repeating sections and add + // them + if (data?.levelup?.repeating) { + const profession = data.profession.values.profession; + const costs = getCostsFromData(profession); + for (const index in data.levelup.repeating) { + // Need to extract the skillname so I can call addRepeating again + const prefix = data.levelup.repeating[index]; + // See if it's spell list. + if (prefix.includes("spelllist")) { + savedspelllists.push(prefix); + continue; + } + if (prefix.endsWith("_talent")) { + savedtalents.push(prefix); + continue; + } + if (prefix.endsWith("_statgain")) { + savedstatgains.push(prefix); + continue; + } + const match = prefix.match(/cmskill_(.*)_specializations/) + if (!match || match.length < 2) { + rmuerror("Coulnd't match regex against ", prefix, index); + continue; + } + const origname = match[0]; + const skillname = match[1]; + const skill = RMUSkills.getSkillByName(skillname) + const costarray = generateCostArray(costs[skill.category]); + addRepeatingSection(`cmskill${skillname}`, 'specialization', origname, + (sectionname) => { console.log("recreate section", sectionname); const sname = `${sectionname}_specname`; const rname = `${sectionname}_cmskillranks`; cmPopulateListOptions(rname, costarray, data.levelup.values[rname]) setAttrsPending({[sname]: data.levelup.values[sname]}); - } - ); - } - } + } + ); + } + } + }); addPendingFunction('Talents add', () => { + const data = getCharmancerData(); for (obj of savedtalents) { addRepeatingSection('added_talents', 'talent', (id) => { setCharmancerOptions(`${id}_talent`, "Category:Talent", @@ -8314,6 +8349,7 @@

Recalculate

}); addPendingFunction('Spell List add', () => { + const data = getCharmancerData(); getAttrsPending(["realm"], (realm) => { initSpellListsCM( savedspelllists, @@ -8694,7 +8730,7 @@

Recalculate

highestval = statval; } else if (highestval == statval) { highest.push(st); - } else if (statval > secondhighest) { + } else if (statval > secondhighestval) { secondhighest = []; secondhighest.push(st) secondhighestval = statval; @@ -9551,6 +9587,35 @@

Recalculate

return str.replace(/[^:]*:/, "").replace(/\?.*/, ""); } +function saveCostData(pdata, kdata) { + const updates = {}; + console.log("save cost data:", pdata, kdata); + { + const scosts = pdata['data-skillCost']; + for (const [group, cost] of Object.entries(scosts)) { + updates[RMUSkills.getCatByDisplayName(group)?.aname + "_cost"] = cost; + } + } + { + const mcosts = pdata['data-Spellcasting']; + updates.magicalritual_cost = mcosts['Magic Ritual']; + updates.spellownbase_cost = mcosts['Base']; + updates.spellopen_cost = mcosts['Open']; + updates.spellclosed_cost = mcosts['Closed']; + updates.spellrestricted_cost = mcosts['Restricted']; + updates.spellarcane_cost = mcosts['Arcane']; + } + { + const wcosts = getWeaponCosts(kdata); + for (const [group, cost] of Object.entries(wcosts)){ + updates[`${group}_cost`] = cost; + } + } + updates.costdatasaved = true; + console.log(updates); + setAttrsPending(updates); +} + /**/ onCheck("mancerfinish:newcharacter", (ev) => { const data = ev.data; @@ -9566,6 +9631,7 @@

Recalculate

if (realm == 'None') { realm = cleanCompendiumName(data.profession.values.realm); } + saveCostData(data.profession.data.profession, data.knacks.values); setAttrsPending({profession: cleanCompendiumName(data.profession.values.profession), realm: realm, xp: 10000, @@ -9665,13 +9731,7 @@

Recalculate

addPendingFunction("CMFinish: Save Weapon Costs", () => { - console.log("Saving", data.knacks.values) - const costs = getWeaponCosts(data.knacks.values); - const updates = {} - for (const [group, cost] of Object.entries(costs)){ - updates[`${group}_cost`] = cost; - } - setAttrsPending(updates); + }); addPendingFunction("CMFinish: update all skills", RMUSkills.updateAllSkills); diff --git a/RolemasterUnified_Official/sheet.json b/RolemasterUnified_Official/sheet.json index 0cff76fb228..497cefcf0b4 100644 --- a/RolemasterUnified_Official/sheet.json +++ b/RolemasterUnified_Official/sheet.json @@ -8,5 +8,5 @@ "legacy": false, "printable": true, "compendium": "RMU", - "version": "1723745861" + "version": "1724166334" } diff --git a/RolemasterUnified_Official/updates.md b/RolemasterUnified_Official/updates.md index 5e111f907dc..7dbb45a3203 100644 --- a/RolemasterUnified_Official/updates.md +++ b/RolemasterUnified_Official/updates.md @@ -1,3 +1,14 @@ +# 2024-8-20 + +- Characters now save their skill costs on creation + - Enables use of custom professions + - Helps with Roll20 Characters (create characters outside of game) +- Support: Add "charactermancer" button to the settings page. + - Will break your harcater. You hae been warned + - Allows migration to new skill costs +- Bugfix: Highest stat gets confused if the first is highest, and the second is not the + secondhighest. Add tests for these cases. + # 2024-8-15 - Show BMR info on front page