Module:GameData/doc: Difference between revisions
From Melvor Idle
(Update JS) |
(Update JS - Use localization data for modifier descriptions & slayer task categories) |
||
(7 intermediate revisions by the same user not shown) | |||
Line 15: | Line 15: | ||
this.debugMode = false; | this.debugMode = false; | ||
this.prettyPrint = false; | this.prettyPrint = false; | ||
this. | this.customLocalizations = { | ||
// Contains custom localization strings, to override any game provided localizations. | |||
// To be used sparingly, for instances where 2+ objects of the same type | |||
// (e.g. monsters) have the same name, as this isn't convenient to deal with in Lua | |||
// TotH curse also named 'Madness' | |||
MAGIC_ABYSSAL_NAME_Madness: 'Madness (ItA)', | |||
// Stronghold boss monsters, where names overlap with normal monster variants | |||
MONSTER_NAME_FierceDevilBoss: 'Fierce Devil (Stronghold)', | |||
MONSTER_NAME_ElementalistBoss: 'Elementalist (Stronghold)', | |||
MONSTER_NAME_PratTheGuardianOfSecretsBoss: 'Prat, the Guardian of Secrets (Stronghold)', | |||
MONSTER_NAME_MysteriousFigurePhase1Stronghold: 'Mysterious Figure - Phase 1 (Stronghold)', | |||
MONSTER_NAME_MysteriousFigurePhase2Stronghold: 'Mysterious Figure - Phase 2 (Stronghold)', | |||
MONSTER_NAME_AhreniaStronghold: 'Ahrenia (Stronghold)' | |||
}; | |||
this.namespaces = { | this.namespaces = { | ||
melvorD: { | melvorD: { | ||
Line 42: | Line 57: | ||
}, | }, | ||
}; | }; | ||
this.registeredNamespaces = []; | |||
// List of categories to be excluded from the generated game data. | |||
// These serve no purpose for the wiki and so would otherwise bloat the data | |||
this.excludedCategories = [ | |||
'pages', | |||
'steamAchievements', | |||
'tutorialStageOrder', | |||
'tutorialStages' | |||
]; | |||
// Check all required namespaces are registered, as there are still some bits of data extracted from in-game rather than the data packages | // Check all required namespaces are registered, as there are still some bits of data extracted from in-game rather than the data packages | ||
Object.keys(this.namespaces).forEach((nsID) => { | Object.keys(this.namespaces).forEach((nsID) => { | ||
Line 93: | Line 117: | ||
} | } | ||
getDataPackURL(nsID) { | getDataPackURL(nsID) { | ||
return 'https://' + location.hostname + | return 'https://' + location.hostname + '/assets/data/' + this.namespaces[nsID].packFile + '?' + DATA_VERSION.toString(); | ||
} | } | ||
async getWikiData() { | async getWikiData() { | ||
Line 117: | Line 141: | ||
console.log(`After transformation: ${JSON.stringify(dataPackage.data).length.toLocaleString()} bytes`); | console.log(`After transformation: ${JSON.stringify(dataPackage.data).length.toLocaleString()} bytes`); | ||
} | } | ||
// Process dependent data after all packages processed | |||
console.log('Processing dependent data for all packages...'); | |||
this.processDependentData(); | |||
// Apply modifications that are to be performed across all packages | |||
this.transformConsolidatedData(); | |||
// All data packages should now be within this.gameData | // All data packages should now be within this.gameData | ||
} | } | ||
Line 250: | Line 279: | ||
const packData = dataPackage.data; | const packData = dataPackage.data; | ||
Object.keys(packData). | Object.keys(packData) | ||
.filter((categoryName) => !this.excludedCategories.includes(categoryName)) | |||
.forEach((categoryName) => { | |||
this.transformDataNode(ns, categoryName, packData, categoryName); | |||
}); | |||
} | } | ||
transformDataNode(ns, categoryName, parentNode, nodeKey) { | transformDataNode(ns, categoryName, parentNode, nodeKey) { | ||
Line 306: | Line 325: | ||
// Special case for skillData so that certain values initialized when the various Skill | // Special case for skillData so that certain values initialized when the various Skill | ||
// classes are initialized may be added here also | // classes are initialized may be added here also | ||
if (categoryName === 'skillData' && | const curSkillID = dataNode.skillID; | ||
if (categoryName === 'skillData' && curSkillID !== undefined && dataNode.data !== undefined) { | |||
// We are currently at the topmost level of a skill object | // We are currently at the topmost level of a skill object | ||
const gameSkill = game.skills.getObjectByID( | const gameSkill = game.skills.getObjectByID(curSkillID); | ||
// For every skill with mastery, add mastery checkpoint descriptions | // For every skill with mastery, add mastery checkpoint descriptions | ||
if ( | if ( | ||
Line 315: | Line 335: | ||
dataNode.data.masteryCheckpoints === undefined | dataNode.data.masteryCheckpoints === undefined | ||
) { | ) { | ||
const localID = this.getLocalID( | const localID = this.getLocalID(curSkillID); | ||
dataNode.data.baseMasteryPoolCap = gameSkill.baseMasteryPoolCap; | dataNode.data.baseMasteryPoolCap = gameSkill.baseMasteryPoolCap; | ||
dataNode.data.masteryCheckpoints = []; | dataNode.data.masteryCheckpoints = []; | ||
Line 322: | Line 342: | ||
}); | }); | ||
} | } | ||
if (!this.skillDataInit[ | if (!this.skillDataInit[curSkillID]) { | ||
if (gameSkill !== undefined) { | if (gameSkill !== undefined) { | ||
// Import other attributes varying by skill | // Import other attributes varying by skill | ||
let importKeys = []; | let importKeys = []; | ||
switch ( | switch (curSkillID) { | ||
case 'melvorD:Mining': | case 'melvorD:Mining': | ||
importKeys = ['baseInterval', 'baseRockHP', 'passiveRegenInterval']; | importKeys = ['baseInterval', 'baseRockHP', 'passiveRegenInterval']; | ||
Line 397: | Line 417: | ||
} | } | ||
} | } | ||
this.skillDataInit[ | this.skillDataInit[curSkillID] = true; | ||
} | } | ||
// Appy localization (skills) | // Appy localization (skills) | ||
Line 403: | Line 423: | ||
} | } | ||
} | } | ||
// Applies any transformations that must be performed on the consolidated gameData | |||
// | // rather than on a per package basis | ||
transformConsolidatedData() { | |||
// Firemaking - Apply default primary & secondary products to logs | |||
const skillDataFM = this.getObjectByID(this.gameData.skillData, 'melvorD:Firemaking', 'skillID'); | |||
skillDataFM.data.logs.forEach((log) => { | |||
} | if (log.primaryProducts === undefined) { | ||
// | log.primaryProducts = [...skillDataFM.data.defaultPrimaryProducts]; | ||
this. | } | ||
// | if (log.secondaryProducts === undefined) { | ||
Object.keys(packData).forEach((categoryName) => { | log.secondaryProducts = [...skillDataFM.data.defaultSecondaryProducts]; | ||
} | |||
}); | |||
// Cooking - Generate perfect fish item downgrades | |||
const skillDataCooking = this.getObjectByID(this.gameData.skillData, 'melvorD:Cooking', 'skillID'); | |||
// Logic sourced from postDataRegistration() within cooking.js | |||
skillDataCooking.data.recipes.forEach((recipe) => { | |||
const perfectID = recipe.perfectCookID; | |||
if (perfectID !== undefined) { | |||
this.gameData.itemUpgrades.push({ | |||
currencyCosts: [], | |||
itemCosts: [ | |||
{ | |||
id: perfectID, | |||
quantity: 1 | |||
} | |||
], | |||
rootItemIDs: [ | |||
perfectID | |||
], | |||
upgradedItemID: recipe.productID, | |||
isDowngrade: true, | |||
}); | |||
} | |||
}); | |||
// Herblore - Generate potion tier item upgrades | |||
const skillDataHerblore = this.getObjectByID(this.gameData.skillData, 'melvorD:Herblore', 'skillID'); | |||
// Logic sourced from postDataRegistration() within herblore.js | |||
skillDataHerblore.data.recipes.forEach((recipe) => { | |||
for (let i = 0; i < recipe.potionIDs.length - 1; i++) { | |||
this.gameData.itemUpgrades.push({ | |||
currencyCosts: [], | |||
itemCosts: [ | |||
{ | |||
id: recipe.potionIDs[i], | |||
quantity: 3, | |||
} | |||
], | |||
rootItemIDs: [ | |||
recipe.potionIDs[i] | |||
], | |||
upgradedItemID: recipe.potionIDs[i + 1], | |||
isDowngrade: false, | |||
}); | |||
} | |||
}); | |||
} | |||
registerPackData(packData) { | |||
Object.keys(packData) | |||
.filter((categoryName) => !this.excludedCategories.includes(categoryName)) | |||
.forEach((categoryName) => { | |||
let categoryData = packData[categoryName]; | let categoryData = packData[categoryName]; | ||
// Some data is adjusted before combining - do this here | // Some data is adjusted before combining - do this here | ||
if (['combatAreas', 'dungeons', 'slayerAreas', 'abyssDepths'].includes(categoryName)) { | if (['combatAreas', 'dungeons', 'slayerAreas', 'abyssDepths', 'strongholds'].includes(categoryName)) { | ||
// Add area type to each area object | // Add area type to each area object | ||
const areaTypes = { | const areaTypes = { | ||
Line 422: | Line 495: | ||
dungeons: 'dungeon', | dungeons: 'dungeon', | ||
slayerAreas: 'slayerArea', | slayerAreas: 'slayerArea', | ||
strongholds: 'stronghold', | |||
abyssDepths: 'abyssDepth', | abyssDepths: 'abyssDepth', | ||
}; | }; | ||
Line 428: | Line 502: | ||
newData.forEach((x) => (x.type = areaType)); | newData.forEach((x) => (x.type = areaType)); | ||
categoryData = newData; | categoryData = newData; | ||
} | } | ||
else if (categoryName === 'golbinRaid') { | |||
} | } | ||
// Data must be pushed into the consoldiated data, rules for vary | // Data must be pushed into the consoldiated data, rules for vary | ||
Line 594: | Line 660: | ||
} | } | ||
}); | }); | ||
} | } | ||
registerDataPackage(namespace) { | |||
// | // Consolidates the data package identified by namespace with existing data within | ||
// | // this.gameData | ||
const packData = this.packData[namespace].data; | |||
if (packData === undefined) { | |||
// | throw new Error(`Couldn't find data for package ${namespace}`); | ||
} | |||
// Add data within the game but outside of data packs | |||
this.registerNonPackData(); | |||
// Consolidate data | |||
this.registerPackData(packData); | |||
// If the data package contains modifications, apply these also | |||
const modificationData = this.packData[namespace].modifications; | |||
if (modificationData !== undefined) { | |||
this.applyDataModifications(modificationData); | |||
} | |||
// Dependent data is handled later, once all packages have been registered | |||
if (!this.registeredNamespaces.includes(namespace)) { | |||
const | this.registeredNamespaces.push(namespace); | ||
if ( | } | ||
} | |||
processDependentData() { | |||
Object.entries(this.packData) | |||
.forEach(([namespace, packData]) => { | |||
if (packData.dependentData !== undefined) { | |||
packData.dependentData.forEach((depDataForNS) => { | |||
const depNS = depDataForNS.namespace; | |||
if (!this.registeredNamespaces.includes(depNS)) { | |||
console.warn( | console.warn( | ||
`Could not apply data | `Could not apply dependent data from package ${namespace}: Data depends on namespace ${depNS}, which has not been registered` | ||
); | ); | ||
} | } | ||
else { | |||
console.log(`Attempting to apply dependent data for ${depNS} from package ${namespace}`); | |||
if (depDataForNS.data !== undefined) { | |||
this.registerPackData(depDataForNS.data) | |||
} | |||
if (depDataForNS.modifications !== undefined) { | |||
this.applyDataModifications(depDataForNS.modifications); | |||
} | |||
console. | |||
. | |||
} | } | ||
} | }); | ||
}); | } | ||
}); | |||
} | |||
getDataToModify(modCat) { | |||
switch (modCat) { | |||
case 'combatAreaCategories': | |||
case 'damageTypes': | |||
case 'dungeons': | |||
case 'equipmentSlots': | |||
) | case 'gamemodes': | ||
case 'items': | |||
case 'modifiers': | |||
case 'pets': | |||
case 'shopUpgradeChains': | |||
case 'shopPurchases': | |||
case 'skillData': | |||
case 'slayerAreas': | |||
return this.gameData[modCat]; | |||
case 'cookingCategories': | |||
const cookingSkill = this.getObjectByID(this.gameData.skillData, 'melvorD:Cooking', 'skillID'); | |||
return cookingSkill.data.categories; | |||
case 'fletchingRecipes': | |||
const fletchingSkill = this.getObjectByID(this.gameData.skillData, 'melvorD:Fletching', 'skillID'); | |||
return fletchingSkill.data.recipes; | |||
} | |||
return undefined; | |||
} | |||
applyModifierModifications(objToModify, adjustments) { | |||
if (objToModify.modifiers === undefined) { | |||
objToModify.modifiers = {}; | |||
} | |||
Object.keys(adjustments) | |||
.forEach((adjType) => { | |||
if (adjType === 'add') { | |||
Object.entries(adjustments[adjType]) | |||
.forEach(([chgKey, chgVal]) => { | |||
if (objToModify.modifiers[chgKey] === undefined) { | |||
objToModify.modifiers[chgKey] = chgVal; | |||
} | |||
else if (Array.isArray(chgVal)) { | |||
objToModify.modifiers[chgKey].push(...chgVal); | |||
} | |||
else { | |||
objToModify.modifiers[chgKey] += chgVal; | |||
} | |||
}); | |||
} | |||
else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled modifier adjustment "${adjType}"` | |||
); | |||
} | |||
} | |||
); | |||
} | |||
applyAddRemoveModifications(objToModify, adjustments, modifyKey) { | |||
if (adjustments.remove !== undefined && Array.isArray(objToModify[modifyKey])) { | |||
// adjustments.remove is an array of requirement types to be removed | |||
let i = 0; | |||
while (i < objToModify[modifyKey].length) { | |||
if (adjustments.remove.includes(objToModify[modifyKey][i].type)) { | |||
objToModify[modifyKey].splice(i, 1); | |||
} | |||
else { | |||
i++; | |||
} | |||
} | |||
} | |||
if (adjustments.add !== undefined) { | |||
if (objToModify[modifyKey] === undefined) { | |||
objToModify[modifyKey] = adjustments.add; | |||
} | |||
else { | |||
objToModify[modifyKey].push(...adjustments.add); | |||
} | |||
} | |||
} | |||
applyGamemodeSpecificModifications(objToModify, adjustments, newProperty) { | |||
const gamemodeID = adjustments.gamemodeID; | |||
if (objToModify.gamemodeOverrides === undefined) { | |||
objToModify.gamemodeOverrides = []; | |||
} | |||
let gamemodeEntryToModify = this.getObjectByID(objToModify.gamemodeOverrides, gamemodeID, 'gamemodeID'); | |||
if (gamemodeEntryToModify === undefined) { | |||
// Initialize gamemode overrides | |||
objToModify.gamemodeOverrides.push({ | |||
gamemodeID: gamemodeID | |||
}); | |||
gamemodeEntryToModify = this.getObjectByID(objToModify.gamemodeOverrides, gamemodeID, 'gamemodeID'); | |||
} | |||
if (gamemodeEntryToModify[newProperty] === undefined) { | |||
gamemodeEntryToModify[newProperty] = structuredClone(objToModify[newProperty]) ?? {}; | |||
} | |||
this.applyAddRemoveModifications(gamemodeEntryToModify, adjustments, newProperty); | |||
} | |||
applyDataModifications(modData) { | |||
const modDataKeys = Object.keys(modData).filter((modCatID) => !this.excludedCategories.includes(modCatID)); | |||
for (const modCatID in modDataKeys) { | |||
const modCat = modDataKeys[modCatID]; | |||
const catData = modData[modCat]; | |||
const dataToModify = this.getDataToModify(modCat); | |||
const modObjIDKey = (modCat === 'skillData' ? 'skillID' : 'id'); | |||
if (dataToModify === undefined) { | |||
console.warn( | |||
`Could not apply data modification for category "${modCat}": Unable to retrieve category data to be modified` | |||
); | |||
} | |||
else { | |||
catData.forEach((modItem) => { | catData.forEach((modItem) => { | ||
const modObjID = modItem | const modObjID = modItem[modObjIDKey]; | ||
if (modObjID === undefined) { | if (modObjID === undefined) { | ||
console.warn( | console.warn( | ||
`Could not apply data modification: ID of object to be modified not found | `Could not apply data modification for category "${modCat}": ID of object to be modified not found` | ||
); | ); | ||
} else { | } else { | ||
const | const objToModify = this.getObjectByID(dataToModify, modObjID, modObjIDKey); | ||
if ( | if (objToModify === undefined) { | ||
console.warn( | console.warn( | ||
`Could not apply data modification: Object with ID "${modObjID}" not found for | `Could not apply data modification: Object with ID "${modObjID}" not found for ctaegory "${modCat}"` | ||
); | ); | ||
} | } | ||
else { | |||
switch (modCat) { | |||
case 'combatAreaCategories': | |||
// The 'areas' property of elements within the category data are ordered data | |||
objToModify.areas = this.combineOrderedData(objToModify.areas, modItem.areas.add); | |||
break; | |||
case 'damageTypes': | |||
Object.entries(modItem) | |||
.filter(([k, v]) => k !== 'id') | |||
.forEach(([k, v]) => { | |||
if (typeof v === 'object' && (v.add !== undefined || v.remove !== undefined)) { | |||
this.applyAddRemoveModifications(objToModify, v, k); | |||
} | |||
else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled key "${k}" for category "${modCat}", object "${modObjID}"` | |||
); | |||
} | |||
}); | |||
break; | |||
case 'gamemodes': | |||
Object.entries(modItem) | |||
.filter(([k, v]) => k !== 'id') | |||
.forEach(([k, v]) => { | |||
if ( | if (typeof v === 'object' && (v.add !== undefined || v.remove !== undefined)) { | ||
this.applyAddRemoveModifications(objToModify, v, k); | |||
} | |||
else if (['abyssalLevelCapCost', 'post99RollConversion'].includes(k)) { | |||
objToModify[k] = v; | |||
} | |||
else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled key "${k}" for category "${modCat}", object "${modObjID}"` | |||
); | |||
} | |||
}); | |||
break; | |||
case 'shopPurchases': | |||
case 'shopUpgradeChains': | |||
// Modify the root upgrade ID of shop upgrade chains, and modify attributes of shop purchases | |||
const overrideKeys = { | |||
purchaseRequirements: { | |||
sourceKey: 'newRequirements', // Key that holds the data in the data package | |||
destKey: 'purchaseRequirementsOverrides', // Key to insert into within this.gameData | |||
subKey: 'requirements', // Sub-key containing the override data | |||
}, | |||
cost: { | |||
sourceKey: 'newCosts', | |||
destKey: 'costOverrides', | |||
subKey: 'cost', | |||
}, | |||
}; | |||
Object.keys(modItem) | |||
.filter((k) => k !== 'id') | |||
.forEach((k) => { | |||
const overrideKey = overrideKeys[k]; | |||
if (overrideKey !== undefined) { | |||
// Is an override specific to a gamemode, do not replace | |||
// the key's existing data | |||
const destKey = overrideKey.destKey; | |||
if (objToModify[destKey] === undefined) { | |||
objToModify[destKey] = []; | |||
} | } | ||
modItem[k].forEach((gamemodeOverride) => { | |||
var newData = {}; | |||
newData.gamemodeID = gamemodeOverride.gamemodeID; | |||
newData[overrideKey.subKey] = gamemodeOverride[overrideKey.sourceKey]; | |||
objToModify[destKey].push(newData); | |||
}); | }); | ||
} else { | |||
objToModify[k] = modItem[k]; | |||
} | |||
}); | |||
break; | |||
case 'cookingCategories': | |||
// Append to the list of shop upgrade IDs for cooking utilities/categories | |||
case 'fletchingRecipes': | |||
// Append to alternativeCosts property of recipes (e.g. Arrow shafts) | |||
Object.keys(modItem) | |||
.filter((k) => k !== 'id') | |||
.forEach((k) => { | |||
if ((k === 'shopUpgradeIDs') || (k === 'alternativeCosts')) { | |||
if (objToModify[k] === undefined) { | |||
objToModify[k] = modItem[k]; | |||
} else { | |||
objToModify[k].push(...modItem[k]); | |||
} | |||
} else { | } else { | ||
console.warn( | console.warn( | ||
`Could not apply data modification: | `Could not apply data modification: Unhandled key "${k}" for category "${modCat}", object "${modObjID}"` | ||
); | ); | ||
} | } | ||
}); | }); | ||
break; | |||
case 'skillData': | |||
Object.entries(modItem.data) | |||
.forEach(([skillProp, propModData]) => { | |||
propModData.forEach((subModItem) => { | |||
const subObjToModify = this.getObjectByID(objToModify.data[skillProp], subModItem.id); | |||
if (subObjToModify === undefined) { | |||
console.warn(`Couldn't find skill object with ID ${subModItem.id} to modify. Property ${skillProp} in skill ID ${objToModify.skillID}`); | |||
} | |||
else { | |||
Object.entries(subModItem) | |||
.forEach(([subProp, subData]) => { | |||
if (subProp === 'modifiers') { | |||
this.applyModifierModifications(subObjToModify, subData); | |||
} | |||
else if (subProp !== 'id') { | |||
this.applyAddRemoveModifications(subObjToModify, subData, subProp); | |||
} | |||
}); | |||
} | } | ||
}); | |||
}); | |||
break; | |||
case 'dungeons': | |||
// Add gamemode specific data to dungeons | |||
Object.keys(modItem) | |||
.filter((k) => k !== 'id') | |||
.forEach((k) => { | |||
if (k === 'gamemodeRewardItemIDs') { | |||
// Add gamemode specific item rewards to dungeon data | |||
const itemRules = modItem[k]; | |||
Object.keys(itemRules).forEach((ruleKey) => { | |||
if (ruleKey === 'add') { | |||
itemRules[ruleKey].forEach((itemDef) => { | |||
const modToApply = { | |||
gamemodeID: itemDef.gamemodeID, | |||
add: itemDef.rewardItemIDs | |||
} | |||
this.applyGamemodeSpecificModifications(objToModify, modToApply, 'rewardItemIDs'); | |||
}); | |||
} else { | |||
console.warn( | |||
`Could not apply data modification: Unknown rule for gamemode item rewards: "${ruleKey}", object "${modObjID}"` | |||
); | |||
} | |||
}); | |||
} else if (k === 'gamemodeEntryRequirements') { | |||
// Add or remove gamemode specific entry requirements to dungeon data | |||
this.applyGamemodeSpecificModifications(objToModify, modItem[k], 'entryRequirements'); | |||
} else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled key "${k}" for category "${modCat}", object "${modObjID}"` | |||
); | |||
} | } | ||
}); | |||
break; | |||
case 'modifiers': | |||
// Add modifier aliases to existing mod scopes | |||
if (objToModify.allowedScopes === undefined) { | |||
console.warn(`Could not apply data modification: Modifier with ID ${modObjID} not found or modifier has no scopes`); | |||
} else { | |||
modItem.allowedScopes.forEach((srcScope) => { | |||
// Find scope within modifier objToModify with matching scopes definition | |||
const srcScopeKeys = Object.keys(srcScope.scopes); | |||
objToModify.allowedScopes.forEach((destScope) => { | |||
const destScopeKeys = Object.keys(destScope.scopes); | |||
const scopeMatch = ( | |||
srcScopeKeys.length === destScopeKeys.length | |||
&& srcScopeKeys.every((k) => destScope.scopes[k] !== undefined && srcScope.scopes[k] == destScope.scopes[k]) | |||
); | |||
if (scopeMatch) { | |||
// Scopes match - add aliases to modifier allowedScope definition | |||
const aliasKeys = ['posAliases', 'negAliases']; | |||
aliasKeys.forEach((aliasKey) => { | |||
if (srcScope[aliasKey] !== undefined) { | |||
if (destScope[aliasKey] === undefined) { | |||
destScope[aliasKey] = []; | |||
} | |||
destScope[aliasKey].push(...srcScope[aliasKey]); | |||
} | |||
}); | |||
} | |||
}); | |||
}); | }); | ||
} | } | ||
}); | break; | ||
case 'items': | |||
Object.keys(modItem) | |||
.filter((k) => k !== 'id') | |||
.forEach((k) => { | |||
if (k === 'modifiers') { | |||
this.applyModifierModifications(objToModify, modItem[k]); | |||
} | } | ||
} | else if (k === 'consumesOn') { | ||
} | Object.keys(modItem[k]) | ||
registerNonPackData() { | .forEach((adjType) => { | ||
// Some data resides outside of packages. Add any such data to this.gameData within this function | if (adjType === 'add') { | ||
// Metadata for data/file version | if (objToModify[k] === undefined) { | ||
objToModify[k] = modItem[k][adjType]; | |||
} | |||
else { | |||
objToModify[k].push(...modItem[k][adjType]); | |||
} | |||
} | |||
else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled adjustment type "${adjType}" for category "${modCat}", object "${modObjID}, property ${k}"` | |||
); | |||
} | |||
}); | |||
} | |||
else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled key "${k}" for category "${modCat}", object "${modObjID}"` | |||
); | |||
} | |||
} | |||
); | |||
break; | |||
case 'pets': | |||
Object.keys(modItem) | |||
.filter((k) => k !== 'id') | |||
.forEach((k) => { | |||
if (k === 'modifiers') { | |||
this.applyModifierModifications(objToModify, modItem[k]); | |||
} | |||
else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled key "${k}" for category "${modCat}", object "${modObjID}"` | |||
); | |||
} | |||
} | |||
); | |||
break; | |||
case 'equipmentSlots': | |||
Object.keys(modItem) | |||
.filter((k) => k !== 'id') | |||
.forEach((k) => { | |||
if (k === 'requirements') { | |||
this.applyAddRemoveModifications(objToModify, modItem[k], 'requirements'); | |||
} | |||
else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled key "${k}" for category "${modCat}", object "${modObjID}"` | |||
); | |||
} | |||
} | |||
); | |||
break; | |||
case 'slayerAreas': | |||
Object.keys(modItem) | |||
.filter((k) => k !== 'id') | |||
.forEach((k) => { | |||
if (k === 'gamemodeEntryRequirements') { | |||
this.applyGamemodeSpecificModifications(objToModify, modItem[k], 'entryRequirements'); | |||
} | |||
else { | |||
console.warn( | |||
`Could not apply data modification: Unhandled key "${k}" for category "${modCat}", object "${modObjID}"` | |||
); | |||
} | |||
} | |||
); | |||
break; | |||
default: | |||
console.warn( | |||
`Could not apply data modification: Unhandled category "${modCat}"` | |||
); | |||
} | |||
} | |||
} | |||
}); | |||
} | |||
} | |||
} | |||
registerNonPackData() { | |||
// Some data resides outside of packages. Add any such data to this.gameData within this function | |||
// Metadata for data/file version | |||
if (this.gameData._dataVersion === undefined) { | if (this.gameData._dataVersion === undefined) { | ||
this.gameData._dataVersion = ({ | this.gameData._dataVersion = ({ | ||
Line 899: | Line 1,133: | ||
})); | })); | ||
} | } | ||
// Normal damage type | // Normal damage type exists outside of data packages | ||
if (this.gameData.damageTypes === undefined) { | if (this.gameData.damageTypes === undefined) { | ||
this.gameData.damageTypes = game.damageTypes | this.gameData.damageTypes = game.damageTypes | ||
Line 910: | Line 1,144: | ||
})); | })); | ||
} | } | ||
if (this.gameData.combatAreaDifficulties === undefined) { | |||
if (this.gameData.combatAreaDifficulties === undefined) { | |||
this.gameData.combatAreaDifficulties = CombatAreaMenuElement.difficulty.map((i) => i.name); | this.gameData.combatAreaDifficulties = CombatAreaMenuElement.difficulty.map((i) => i.name); | ||
} | } | ||
if (this.gameData.attackTypes === undefined) { | if (this.gameData.attackTypes === undefined) { | ||
this.gameData.attackTypes = AttackTypeID; | this.gameData.attackTypes = AttackTypeID; | ||
} | } | ||
} | } | ||
combineOrderedData(existingData, newData) { | combineOrderedData(existingData, newData) { | ||
Line 1,058: | Line 1,235: | ||
const newStats = {}; | const newStats = {}; | ||
entity.forEach((stat) => { | entity.forEach((stat) => { | ||
if (newStats[ | let statKey = stat.key; | ||
newStats[ | if (stat.damageType !== undefined) { | ||
statKey += this.getLocalID(stat.damageType); | |||
} | |||
if (newStats[statKey] === undefined) { | |||
newStats[statKey] = stat.value; | |||
} else { | } else { | ||
newStats[ | newStats[statKey] += stat.value; | ||
} | } | ||
}); | }); | ||
Line 1,194: | Line 1,375: | ||
const spell = game.attackSpells.getObjectByID(data.id); | const spell = game.attackSpells.getObjectByID(data.id); | ||
if (spell !== undefined) { | if (spell !== undefined) { | ||
return spell. | return this.getLangString(`${ spell.spellbook.spellNameLangPrefix }${ spell.localID }`); | ||
} | } | ||
}; | }; | ||
Line 1,203: | Line 1,384: | ||
'combatAreaDisplayOrder', | 'combatAreaDisplayOrder', | ||
'combatAreaCategoryOrder', | 'combatAreaCategoryOrder', | ||
'combatEffectTables', | |||
'combatEffectTemplates', | 'combatEffectTemplates', | ||
'combatEvents', | 'combatEvents', | ||
Line 1,211: | Line 1,393: | ||
'itemUpgrades', | 'itemUpgrades', | ||
'itmMonsters', | 'itmMonsters', | ||
'randomAbyssalGems', | 'randomAbyssalGems', | ||
'randomFiremakingOils', | 'randomFiremakingOils', | ||
Line 1,218: | Line 1,399: | ||
'randomSuperiorGems', | 'randomSuperiorGems', | ||
'slayerAreaDisplayOrder', | 'slayerAreaDisplayOrder', | ||
'shopCategoryOrder', | 'shopCategoryOrder', | ||
'shopDisplayOrder', | 'shopDisplayOrder', | ||
Line 1,282: | Line 1,462: | ||
}, | }, | ||
strongholds: { | strongholds: { | ||
name: { | name: { idFormat: 'STRONGHOLD_NAME_{ID}' }, | ||
}, | }, | ||
equipmentSlots: { | equipmentSlots: { | ||
Line 1,299: | Line 1,479: | ||
lore: { | lore: { | ||
title: { key: 'LORE', idFormat: 'TITLE_{ID}' }, | title: { key: 'LORE', idFormat: 'TITLE_{ID}' }, | ||
}, | |||
modifiers: { | |||
// Modifier descriptions are buried quite deep within modifier definitions | |||
_handler: 'modifierDesc', | |||
}, | }, | ||
monsters: { | monsters: { | ||
Line 1,326: | Line 1,510: | ||
name: { key: 'SLAYER_AREA', idFormat: 'NAME_{ID}' }, | name: { key: 'SLAYER_AREA', idFormat: 'NAME_{ID}' }, | ||
areaEffectDescription: { key: 'SLAYER_AREA', idFormat: 'EFFECT_{ID}' }, | areaEffectDescription: { key: 'SLAYER_AREA', idFormat: 'EFFECT_{ID}' }, | ||
}, | |||
slayerTaskCategories: { | |||
name: { idFormat: 'SLAYER_TASK_CATEGORY_{ID}' }, | |||
reqText: { idFormat: 'MENU_TEXT_REQUIRES_SLAYER_{ID}' }, | |||
reqToast: { idFormat: 'TOASTS_SLAYER_TASK_REQUIRED_{ID}' }, | |||
unlockText: { idFormat: 'MENU_TEXT_UNLOCK_SLAYER_{ID}' }, | |||
}, | }, | ||
skillData: { | skillData: { | ||
Line 1,347: | Line 1,537: | ||
name: { key: 'POI_NAME_Melvor' }, | name: { key: 'POI_NAME_Melvor' }, | ||
}, | }, | ||
}, | }, | ||
Agility: { | Agility: { | ||
Line 1,522: | Line 1,711: | ||
} | } | ||
dataToTranslate.forEach((tData) => { | dataToTranslate.forEach((tData) => { | ||
Object.keys(langKeyData).forEach((langKey) => { | Object.keys(langKeyData).forEach((langKey) => { | ||
const targetData = langKey === '_root' ? tData : tData[langKey]; | const targetData = langKey === '_root' ? tData : tData[langKey]; | ||
if (targetData !== undefined) { | if (targetData !== undefined) { | ||
const targetArr = Array.isArray(targetData) ? targetData : [targetData]; | const targetArr = Array.isArray(targetData) ? targetData : [targetData]; | ||
targetArr.forEach((target) => { | targetArr.forEach((target) => { | ||
const handlerFunc = langKeyData[langKey]['_handler']; | const handlerFunc = langKeyData[langKey]['_handler']; | ||
if (handlerFunc !== undefined) { | if (handlerFunc !== undefined) { | ||
switch (handlerFunc) { | switch (handlerFunc) { | ||
case 'mapPortals': | case 'mapPortals': | ||
Object.keys(target).forEach((portalKey) => { | Object.keys(target).forEach((portalKey) => { | ||
let portalData = target[portalKey]; | let portalData = target[portalKey]; | ||
const langID = this.getLocalID(portalData.originWorldMap) + '_' + this.getLocalID(portalData.id); | const langID = this.getLocalID(portalData.originWorldMap) + '_' + this.getLocalID(portalData.id); | ||
portalData.name = this.getLangString('POI_NAME', langID); | portalData.name = this.getLangString('POI_NAME', langID); | ||
portalData.description = this.getLangString('POI_DESCRIPTION', langID); | portalData.description = this.getLangString('POI_DESCRIPTION', langID); | ||
}); | |||
break; | |||
case 'cartoMaps': | |||
// Target represents a world map | |||
const mapID = this.getLocalID(target.id); | |||
target.name = this.getLangString('WORLD_MAP_NAME', mapID); | |||
// Process POIs | |||
target.pointsOfInterest.forEach((poi) => { | |||
const langID = mapID + '_' + this.getLocalID(poi.id); | |||
poi.name = this.getLangString('POI_NAME', langID); | |||
poi.description = this.getLangString('POI_DESCRIPTION', langID); | |||
}); | |||
break; | |||
case 'modifierDesc': | |||
// Target represents a modifier definition | |||
target.allowedScopes.forEach((modScope) => { | |||
modScope.descriptions.forEach((modDesc) => { | |||
const langString = this.getLangString(modDesc.lang); | |||
if (langString !== undefined) { | |||
modDesc.text = langString; | |||
} | |||
delete modDesc.lang; | |||
}); | |||
}); | }); | ||
} | } | ||
} else { | } else { | ||
Line 1,708: | Line 1,908: | ||
lookupVal += (lookupVal.length > 0 ? '_' : '') + identifier; | lookupVal += (lookupVal.length > 0 ? '_' : '') + identifier; | ||
} | } | ||
return loadedLangJson[lookupVal]; | return this.customLocalizations[lookupVal] ?? loadedLangJson[lookupVal]; | ||
} | } | ||
getNamespacedID(namespace, ID) { | getNamespacedID(namespace, ID) { |
Latest revision as of 22:50, 13 January 2025
The GameData module is the source of all game data which many other Lua modules rely upon. This module deals with the initial loading of the game data structure, and then enables other modules to access this both via a library of functions (preferred) and in its raw format.
The game data used by the wiki is currently at version V1.3.1 (?12015).
To generate game data, do the following:
- Navigate to https://melvoridle.com within your preferred web browser
- Select any character, the character that is chosen has no impact but you may consider creating a new one as a precaution - the below code is designed to execute without affecting the character, although this is not guaranteed
- Ensure mods are disabled such that the generated data excludes any modded content. If disabling mods, the game should be reloaded first before trying to generate game data
- Open the browser console/developer mode (usually by hitting the F12 key for most browsers)
- Within the browser console, enter the following code then hit enter. If successful, the game data should appear within the console
- Copy the game data & update Module:GameData/data, Module:GameData/data2, Module:GameData/data3 accordingly
ExpandCode |
---|