import { walk_quests } from "./walk_quests"; import { RESOURCE_DIR } from "./index"; import { npc_data, NpcType } from "../src/core/data_formats/parsing/quest/npc_types"; import { get_npc_type, QuestNpc } from "../src/core/data_formats/parsing/quest/QuestNpc"; import { EntityProp, EntityPropType } from "../src/core/data_formats/parsing/quest/properties"; import { get_entity_prop_value, QuestEntity } from "../src/core/data_formats/parsing/quest/Quest"; const prop_cache = new Map(); print_quest_stats(); function print_quest_stats(): void { const type_data: Map = new Map(); walk_quests(`${RESOURCE_DIR}/tethealla_v0.143_quests`, ({ quest }) => { for (const npc of quest.npcs) { const type = get_npc_type(npc); const npcs = type_data.get(type); if (npcs == undefined) { type_data.set(type, [{ npc, quest: quest.name, count: 1 }]); } else { const found = npcs.find(({ npc: npc_2 }) => entities_equal(npc, npc_2, type)); if (found) { found.count++; } else { npcs.push({ npc, quest: quest.name, count: 1 }); } } } }); for (const [type, npcs] of type_data) { const props = get_properties(type); console.log(NpcType[type]); console.log(" cnt " + props.map(col_print_name).join(" ")); for (const { npc, quest, count } of npcs.sort((a, b) => b.count - a.count).slice(0, 5)) { console.log( ` ${count.toString().padStart(3, " ")} ` + props.map(p => col_print_value(npc, p)).join(" ") + ` ${quest}`, ); } } } /** * @returns the entity's properties enriched with many default properties. */ function get_properties(type: NpcType): EntityProp[] { let props = prop_cache.get(type); if (props) { return props; } const data = npc_data(type); props = [ { name: "unknwn", offset: 2, type: EntityPropType.I16, }, { name: "unknwn", offset: 4, type: EntityPropType.I16, }, { name: "clncnt", offset: 6, type: EntityPropType.I16, }, { name: "unknwn", offset: 8, type: EntityPropType.I16, }, { name: "unknwn", offset: 10, type: EntityPropType.I16, }, { name: "scale x", offset: 44, type: EntityPropType.F32, }, { name: "scale y", offset: 48, type: EntityPropType.F32, }, { name: "scale z", offset: 52, type: EntityPropType.F32, }, { name: "unknwn", offset: 68, type: EntityPropType.I16, }, { name: "unknwn", offset: 70, type: EntityPropType.I16, }, ]; outer: for (const npc_prop of data.properties) { for (let i = 0; i < props.length; i++) { const prop = props[i]; if (npc_prop.offset === prop.offset) { props.splice(i, 1, npc_prop); continue outer; } else if (npc_prop.offset < prop.offset) { props.splice(i, 0, npc_prop); continue outer; } } props.push(npc_prop); } return props; } function col_print_name(prop: EntityProp): string { const width = col_width(prop); return prop.name.slice(0, width).padStart(width, " "); } function col_print_value(entity: QuestEntity, prop: EntityProp): string { const value = get_entity_prop_value(entity, prop); const str = prop.type === EntityPropType.F32 || prop.type === EntityPropType.Angle ? value.toFixed(3) : value.toString(); return str.padStart(col_width(prop), " "); } function col_width(prop: EntityProp): number { switch (prop.type) { case EntityPropType.U8: return 3; case EntityPropType.U16: return 5; case EntityPropType.U32: return 10; case EntityPropType.I8: return 4; case EntityPropType.I16: return 6; case EntityPropType.I32: return 11; case EntityPropType.F32: return 10; case EntityPropType.Angle: return 4; } } function entities_equal(a: QuestEntity, b: QuestEntity, type: NpcType): boolean { for (const prop of get_properties(type)) { if (get_entity_prop_value(a, prop) !== get_entity_prop_value(b, prop)) { return false; } } return true; }