769 lines
43 KiB
Lua
769 lines
43 KiB
Lua
-- Export everything from legends mode
|
|
--[[=begin
|
|
|
|
exportlegends
|
|
=============
|
|
Controls legends mode to export data - especially useful to set-and-forget large
|
|
worlds, or when you want a map of every site when there are several hundred.
|
|
|
|
The 'info' option exports more data than is possible in vanilla, to a
|
|
:file:`region-date-legends_plus.xml` file developed to extend
|
|
:forums:`World Viewer <128932>` and other legends utilities.
|
|
|
|
Options:
|
|
|
|
:info: Exports the world/gen info, the legends XML, and a custom XML with more information
|
|
:custom Exports a custom XML with more information
|
|
:sites: Exports all available site maps
|
|
:maps: Exports all seventeen detailed maps
|
|
:all: Equivalent to calling all of the above, in that order
|
|
|
|
=end]]
|
|
|
|
gui = require 'gui'
|
|
local args = {...}
|
|
local vs = dfhack.gui.getCurViewscreen()
|
|
local i = 1
|
|
|
|
local MAPS = {
|
|
"Standard biome+site map",
|
|
"Elevations including lake and ocean floors",
|
|
"Elevations respecting water level",
|
|
"Biome",
|
|
"Hydrosphere",
|
|
"Temperature",
|
|
"Rainfall",
|
|
"Drainage",
|
|
"Savagery",
|
|
"Volcanism",
|
|
"Current vegetation",
|
|
"Evil",
|
|
"Salinity",
|
|
"Structures/fields/roads/etc.",
|
|
"Trade",
|
|
"Nobility and Holdings",
|
|
"Diplomacy",
|
|
}
|
|
|
|
function getItemSubTypeName(itemType, subType)
|
|
if (dfhack.items.getSubtypeCount(itemType)) <= 0 then
|
|
return tostring(-1)
|
|
end
|
|
local subtypename = dfhack.items.getSubtypeDef(itemType, subType)
|
|
if (subtypename == nil) then
|
|
return tostring(-1)
|
|
else
|
|
return tostring(subtypename.name):lower()
|
|
end
|
|
end
|
|
|
|
function findEntity(id)
|
|
for k,v in ipairs(df.global.world.entities.all) do
|
|
if (v.id == id) then
|
|
return v
|
|
end
|
|
end
|
|
return nil
|
|
end
|
|
|
|
function table.contains(table, element)
|
|
for _, value in pairs(table) do
|
|
if value == element then
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
function table.containskey(table, key)
|
|
for value, _ in pairs(table) do
|
|
if value == key then
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
--create an extra legends xml with extra data, by Mason11987 for World Viewer
|
|
function export_more_legends_xml()
|
|
local month = dfhack.world.ReadCurrentMonth() + 1 --days and months are 1-indexed
|
|
local day = dfhack.world.ReadCurrentDay()
|
|
local year_str = string.format('%0'..math.max(5, string.len(''..df.global.cur_year))..'d', df.global.cur_year)
|
|
local date_str = year_str..string.format('-%02d-%02d', month, day)
|
|
|
|
local filename = df.global.world.cur_savegame.save_dir.."-"..date_str.."-legends_plus.xml"
|
|
local file = io.open(filename, 'w')
|
|
if not file then qerror("could not open file: " .. filename) end
|
|
|
|
file:write("<?xml version=\"1.0\" encoding='UTF-8'?>\n")
|
|
file:write("<df_world>\n")
|
|
file:write("<name>"..dfhack.df2utf(dfhack.TranslateName(df.global.world.world_data.name)).."</name>\n")
|
|
file:write("<altname>"..dfhack.df2utf(dfhack.TranslateName(df.global.world.world_data.name,1)).."</altname>\n")
|
|
|
|
file:write("<landmasses>\n")
|
|
for landmassK, landmassV in ipairs(df.global.world.world_data.landmasses) do
|
|
file:write("\t<landmass>\n")
|
|
file:write("\t\t<id>"..landmassV.index.."</id>\n")
|
|
file:write("\t\t<name>"..dfhack.df2utf(dfhack.TranslateName(landmassV.name,1)).."</name>\n")
|
|
file:write("\t\t<coord_1>"..landmassV.min_x..","..landmassV.min_y.."</coord_1>\n")
|
|
file:write("\t\t<coord_2>"..landmassV.max_x..","..landmassV.max_y.."</coord_2>\n")
|
|
file:write("\t</landmass>\n")
|
|
end
|
|
file:write("</landmasses>\n")
|
|
|
|
file:write("<mountain_peaks>\n")
|
|
for mountainK, mountainV in ipairs(df.global.world.world_data.mountain_peaks) do
|
|
file:write("\t<mountain_peak>\n")
|
|
file:write("\t\t<id>"..mountainK.."</id>\n")
|
|
file:write("\t\t<name>"..dfhack.df2utf(dfhack.TranslateName(mountainV.name,1)).."</name>\n")
|
|
file:write("\t\t<coords>"..mountainV.pos.x..","..mountainV.pos.y.."</coords>\n")
|
|
file:write("\t\t<height>"..mountainV.height.."</height>\n")
|
|
file:write("\t</mountain_peak>\n")
|
|
end
|
|
file:write("</mountain_peaks>\n")
|
|
|
|
file:write("<regions>\n")
|
|
for regionK, regionV in ipairs(df.global.world.world_data.regions) do
|
|
file:write("\t<region>\n")
|
|
file:write("\t\t<id>"..regionV.index.."</id>\n")
|
|
file:write("\t\t<coords>")
|
|
for xK, xVal in ipairs(regionV.region_coords.x) do
|
|
file:write(xVal..","..regionV.region_coords.y[xK].."|")
|
|
end
|
|
file:write("</coords>\n")
|
|
file:write("\t</region>\n")
|
|
end
|
|
file:write("</regions>\n")
|
|
|
|
file:write("<underground_regions>\n")
|
|
for regionK, regionV in ipairs(df.global.world.world_data.underground_regions) do
|
|
file:write("\t<underground_region>\n")
|
|
file:write("\t\t<id>"..regionV.index.."</id>\n")
|
|
file:write("\t\t<coords>")
|
|
for xK, xVal in ipairs(regionV.region_coords.x) do
|
|
file:write(xVal..","..regionV.region_coords.y[xK].."|")
|
|
end
|
|
file:write("</coords>\n")
|
|
file:write("\t</underground_region>\n")
|
|
end
|
|
file:write("</underground_regions>\n")
|
|
|
|
file:write("<sites>\n")
|
|
for siteK, siteV in ipairs(df.global.world.world_data.sites) do
|
|
file:write("\t<site>\n")
|
|
for k,v in pairs(siteV) do
|
|
if (k == "id" or k == "civ_id" or k == "cur_owner_id") then
|
|
file:write("\t\t<"..k..">"..tostring(v).."</"..k..">\n")
|
|
elseif (k == "buildings") then
|
|
if (#siteV.buildings > 0) then
|
|
file:write("\t\t<structures>\n")
|
|
for buildingK, buildingV in ipairs(siteV.buildings) do
|
|
file:write("\t\t\t<structure>\n")
|
|
file:write("\t\t\t\t<id>"..buildingV.id.."</id>\n")
|
|
file:write("\t\t\t\t<type>"..df.abstract_building_type[buildingV:getType()]:lower().."</type>\n")
|
|
if (df.abstract_building_type[buildingV:getType()]:lower() ~= "underworld_spire") then
|
|
-- if spire: unk_50 should be name and unk_bc some kind of flag
|
|
file:write("\t\t\t\t<name>"..dfhack.df2utf(dfhack.TranslateName(buildingV.name, 1)).."</name>\n")
|
|
file:write("\t\t\t\t<name2>"..dfhack.df2utf(dfhack.TranslateName(buildingV.name)).."</name2>\n")
|
|
end
|
|
if (buildingV:getType() == df.abstract_building_type.TEMPLE) then
|
|
file:write("\t\t\t\t<deity>"..buildingV.deity.."</deity>\n")
|
|
file:write("\t\t\t\t<religion>"..buildingV.religion.."</religion>\n")
|
|
end
|
|
if (buildingV:getType() == df.abstract_building_type.DUNGEON) then
|
|
file:write("\t\t\t\t<dungeon_type>"..buildingV.dungeon_type.."</dungeon_type>\n")
|
|
end
|
|
for inhabitabntK,inhabitabntV in pairs(buildingV.inhabitants) do
|
|
file:write("\t\t\t\t<inhabitant>"..inhabitabntV.anon_2.."</inhabitant>\n")
|
|
end
|
|
file:write("\t\t\t</structure>\n")
|
|
end
|
|
file:write("\t\t</structures>\n")
|
|
end
|
|
end
|
|
end
|
|
file:write("\t</site>\n")
|
|
end
|
|
file:write("</sites>\n")
|
|
|
|
file:write("<world_constructions>\n")
|
|
for wcK, wcV in ipairs(df.global.world.world_data.constructions.list) do
|
|
file:write("\t<world_construction>\n")
|
|
file:write("\t\t<id>"..wcV.id.."</id>\n")
|
|
file:write("\t\t<name>"..dfhack.df2utf(dfhack.TranslateName(wcV.name,1)).."</name>\n")
|
|
file:write("\t\t<type>"..(df.world_construction_type[wcV:getType()]):lower().."</type>\n")
|
|
file:write("\t\t<coords>")
|
|
for xK, xVal in ipairs(wcV.square_pos.x) do
|
|
file:write(xVal..","..wcV.square_pos.y[xK].."|")
|
|
end
|
|
file:write("</coords>\n")
|
|
file:write("\t</world_construction>\n")
|
|
end
|
|
file:write("</world_constructions>\n")
|
|
|
|
file:write("<artifacts>\n")
|
|
for artifactK, artifactV in ipairs(df.global.world.artifacts.all) do
|
|
file:write("\t<artifact>\n")
|
|
file:write("\t\t<id>"..artifactV.id.."</id>\n")
|
|
if (artifactV.item:getType() ~= -1) then
|
|
file:write("\t\t<item_type>"..tostring(df.item_type[artifactV.item:getType()]):lower().."</item_type>\n")
|
|
if (artifactV.item:getSubtype() ~= -1) then
|
|
file:write("\t\t<item_subtype>"..artifactV.item.subtype.name.."</item_subtype>\n")
|
|
end
|
|
for improvementK,impovementV in pairs(artifactV.item.improvements) do
|
|
if impovementV:getType() == df.improvement_type.WRITING then
|
|
for writingk,writingV in pairs(impovementV["itemimprovement_writingst.anon_1"]) do
|
|
file:write("\t\t<writing>"..writingV.."</writing>\n")
|
|
end
|
|
elseif impovementV:getType() == df.improvement_type.PAGES then
|
|
file:write("\t\t<page_count>"..impovementV.count.."</page_count>\n")
|
|
for writingk,writingV in pairs(impovementV.contents) do
|
|
file:write("\t\t<writing>"..writingV.."</writing>\n")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
if (table.containskey(artifactV.item,"description")) then
|
|
file:write("\t\t<item_description>"..dfhack.df2utf(artifactV.item.description:lower()).."</item_description>\n")
|
|
end
|
|
if (artifactV.item:getMaterial() ~= -1 and artifactV.item:getMaterialIndex() ~= -1) then
|
|
file:write("\t\t<mat>"..dfhack.matinfo.toString(dfhack.matinfo.decode(artifactV.item:getMaterial(), artifactV.item:getMaterialIndex())).."</mat>\n")
|
|
end
|
|
file:write("\t</artifact>\n")
|
|
end
|
|
file:write("</artifacts>\n")
|
|
|
|
file:write("<historical_figures>\n")
|
|
for hfK, hfV in ipairs(df.global.world.history.figures) do
|
|
file:write("\t<historical_figure>\n")
|
|
file:write("\t\t<id>"..hfV.id.."</id>\n")
|
|
file:write("\t\t<sex>"..hfV.sex.."</sex>\n")
|
|
if hfV.race >= 0 then file:write("\t\t<race>"..df.global.world.raws.creatures.all[hfV.race].name[0].."</race>\n") end
|
|
file:write("\t</historical_figure>\n")
|
|
end
|
|
file:write("</historical_figures>\n")
|
|
|
|
file:write("<entity_populations>\n")
|
|
for entityPopK, entityPopV in ipairs(df.global.world.entity_populations) do
|
|
file:write("\t<entity_population>\n")
|
|
file:write("\t\t<id>"..entityPopV.id.."</id>\n")
|
|
for raceK, raceV in ipairs(entityPopV.races) do
|
|
local raceName = (df.global.world.raws.creatures.all[raceV].creature_id):lower()
|
|
file:write("\t\t<race>"..raceName..":"..entityPopV.counts[raceK].."</race>\n")
|
|
end
|
|
file:write("\t\t<civ_id>"..entityPopV.civ_id.."</civ_id>\n")
|
|
file:write("\t</entity_population>\n")
|
|
end
|
|
file:write("</entity_populations>\n")
|
|
|
|
file:write("<entities>\n")
|
|
for entityK, entityV in ipairs(df.global.world.entities.all) do
|
|
file:write("\t<entity>\n")
|
|
file:write("\t\t<id>"..entityV.id.."</id>\n")
|
|
if entityV.race >= 0 then
|
|
file:write("\t\t<race>"..(df.global.world.raws.creatures.all[entityV.race].creature_id):lower().."</race>\n")
|
|
end
|
|
file:write("\t\t<type>"..(df.historical_entity_type[entityV.type]):lower().."</type>\n")
|
|
if entityV.type == df.historical_entity_type.Religion then -- Get worshipped figure
|
|
if (entityV.unknown1b ~= nil and entityV.unknown1b.worship ~= nil) then
|
|
for k,v in pairs(entityV.unknown1b.worship) do
|
|
file:write("\t\t<worship_id>"..v.."</worship_id>\n")
|
|
end
|
|
end
|
|
end
|
|
for id, link in pairs(entityV.entity_links) do
|
|
file:write("\t\t<entity_link>\n")
|
|
for k, v in pairs(link) do
|
|
if (k == "type") then
|
|
file:write("\t\t\t<"..k..">"..tostring(df.entity_entity_link_type[v]).."</"..k..">\n")
|
|
else
|
|
file:write("\t\t\t<"..k..">"..v.."</"..k..">\n")
|
|
end
|
|
end
|
|
file:write("\t\t</entity_link>\n")
|
|
end
|
|
for positionK,positionV in pairs(entityV.positions.own) do
|
|
file:write("\t\t<entity_position>\n")
|
|
file:write("\t\t\t<id>"..positionV.id.."</id>\n")
|
|
if positionV.name[0] ~= "" then file:write("\t\t\t<name>"..positionV.name[0].."</name>\n") end
|
|
if positionV.name_male[0] ~= "" then file:write("\t\t\t<name_male>"..positionV.name_male[0].."</name_male>\n") end
|
|
if positionV.name_female[0] ~= "" then file:write("\t\t\t<name_female>"..positionV.name_female[0].."</name_female>\n") end
|
|
if positionV.spouse[0] ~= "" then file:write("\t\t\t<spouse>"..positionV.spouse[0].."</spouse>\n") end
|
|
if positionV.spouse_male[0] ~= "" then file:write("\t\t\t<spouse_male>"..positionV.spouse_male[0].."</spouse_male>\n") end
|
|
if positionV.spouse_female[0] ~= "" then file:write("\t\t\t<spouse_female>"..positionV.spouse_female[0].."</spouse_female>\n") end
|
|
file:write("\t\t</entity_position>\n")
|
|
end
|
|
for assignmentK,assignmentV in pairs(entityV.positions.assignments) do
|
|
file:write("\t\t<entity_position_assignment>\n")
|
|
for k, v in pairs(assignmentV) do
|
|
if (k == "id" or k == "histfig" or k == "position_id" or k == "squad_id") then
|
|
file:write("\t\t\t<"..k..">"..v.."</"..k..">\n")
|
|
end
|
|
end
|
|
file:write("\t\t</entity_position_assignment>\n")
|
|
end
|
|
for idx,id in pairs(entityV.histfig_ids) do
|
|
file:write("\t\t<histfig_id>"..id.."</histfig_id>\n")
|
|
end
|
|
for id, link in ipairs(entityV.children) do
|
|
file:write("\t\t<child>"..link.."</child>\n")
|
|
end
|
|
file:write("\t\t<claims>")
|
|
for xK, xVal in ipairs(entityV.claims.unk2.x) do
|
|
file:write(xVal..","..entityV.claims.unk2.y[xK].."|")
|
|
end
|
|
file:write("</claims>\n")
|
|
file:write("\t</entity>\n")
|
|
end
|
|
file:write("</entities>\n")
|
|
|
|
file:write("<poetic_forms>\n")
|
|
for formK, formV in ipairs(df.global.world.poetic_forms.all) do
|
|
file:write("\t<poetic_form>\n")
|
|
file:write("\t\t<id>"..formV.id.."</id>\n")
|
|
file:write("\t\t<name>"..dfhack.df2utf(dfhack.TranslateName(formV.name,1)).."</name>\n")
|
|
file:write("\t</poetic_form>\n")
|
|
end
|
|
file:write("</poetic_forms>\n")
|
|
|
|
file:write("<musical_forms>\n")
|
|
for formK, formV in ipairs(df.global.world.musical_forms.all) do
|
|
file:write("\t<musical_form>\n")
|
|
file:write("\t\t<id>"..formV.id.."</id>\n")
|
|
file:write("\t\t<name>"..dfhack.df2utf(dfhack.TranslateName(formV.name,1)).."</name>\n")
|
|
file:write("\t</musical_form>\n")
|
|
end
|
|
file:write("</musical_forms>\n")
|
|
|
|
file:write("<dance_forms>\n")
|
|
for formK, formV in ipairs(df.global.world.dance_forms.all) do
|
|
file:write("\t<dance_form>\n")
|
|
file:write("\t\t<id>"..formV.id.."</id>\n")
|
|
file:write("\t\t<name>"..dfhack.df2utf(dfhack.TranslateName(formV.name,1)).."</name>\n")
|
|
file:write("\t</dance_form>\n")
|
|
end
|
|
file:write("</dance_forms>\n")
|
|
|
|
file:write("<written_contents>\n")
|
|
for wcK, wcV in ipairs(df.global.world.written_contents.all) do
|
|
file:write("\t<written_content>\n")
|
|
file:write("\t\t<id>"..wcV.id.."</id>\n")
|
|
file:write("\t\t<title>"..wcV.title.."</title>\n")
|
|
file:write("\t\t<page_start>"..wcV.page_start.."</page_start>\n")
|
|
file:write("\t\t<page_end>"..wcV.page_end.."</page_end>\n")
|
|
for refK, refV in pairs(wcV.refs) do
|
|
file:write("\t\t<reference>\n")
|
|
file:write("\t\t\t<type>"..df.general_ref_type[refV:getType()].."</type>\n")
|
|
if refV:getType() == df.general_ref_type.ARTIFACT then file:write("\t\t\t<id>"..refV.artifact_id.."</id>\n") -- artifact
|
|
elseif refV:getType() == df.general_ref_type.ENTITY then file:write("\t\t\t<id>"..refV.entity_id.."</id>\n") -- entity
|
|
elseif refV:getType() == df.general_ref_type.HISTORICAL_EVENT then file:write("\t\t\t<id>"..refV.event_id.."</id>\n") -- event
|
|
elseif refV:getType() == df.general_ref_type.SITE then file:write("\t\t\t<id>"..refV.site_id.."</id>\n") -- site
|
|
elseif refV:getType() == df.general_ref_type.SUBREGION then file:write("\t\t\t<id>"..refV.region_id.."</id>\n") -- region
|
|
elseif refV:getType() == df.general_ref_type.HISTORICAL_FIGURE then file:write("\t\t\t<id>"..refV.hist_figure_id.."</id>\n") -- hist figure
|
|
elseif refV:getType() == df.general_ref_type.WRITTEN_CONTENT then file:write("\t\t\t<id>"..refV.anon_1.."</id>\n")
|
|
elseif refV:getType() == df.general_ref_type.POETIC_FORM then file:write("\t\t\t<id>"..refV.poetic_form_id.."</id>\n") -- poetic form
|
|
elseif refV:getType() == df.general_ref_type.MUSICAL_FORM then file:write("\t\t\t<id>"..refV.musical_form_id.."</id>\n") -- musical form
|
|
elseif refV:getType() == df.general_ref_type.DANCE_FORM then file:write("\t\t\t<id>"..refV.dance_form_id.."</id>\n") -- dance form
|
|
elseif refV:getType() == df.general_ref_type.INTERACTION then -- TODO INTERACTION
|
|
elseif refV:getType() == df.general_ref_type.KNOWLEDGE_SCHOLAR_FLAG then -- TODO KNOWLEDGE_SCHOLAR_FLAG
|
|
elseif refV:getType() == df.general_ref_type.VALUE_LEVEL then -- TODO VALUE_LEVEL
|
|
elseif refV:getType() == df.general_ref_type.LANGUAGE then -- TODO LANGUAGE
|
|
else
|
|
print("unknown reference",refV:getType(),df.general_ref_type[refV:getType()])
|
|
--for k,v in pairs(refV) do print(k,v) end
|
|
end
|
|
file:write("\t\t</reference>\n")
|
|
end
|
|
file:write("\t\t<type>"..(df.written_content_type[wcV.type] or wcV.type).."</type>\n")
|
|
for styleK, styleV in pairs(wcV.styles) do
|
|
file:write("\t\t<style>"..(df.written_content_style[styleV] or styleV).."</style>\n")
|
|
end
|
|
file:write("\t\t<author>"..wcV.author.."</author>\n")
|
|
file:write("\t</written_content>\n")
|
|
end
|
|
file:write("</written_contents>\n")
|
|
|
|
file:write("<historical_events>\n")
|
|
for ID, event in ipairs(df.global.world.history.events) do
|
|
if event:getType() == df.history_event_type.ADD_HF_ENTITY_LINK
|
|
or event:getType() == df.history_event_type.ADD_HF_SITE_LINK
|
|
or event:getType() == df.history_event_type.ADD_HF_HF_LINK
|
|
or event:getType() == df.history_event_type.ADD_HF_ENTITY_LINK
|
|
or event:getType() == df.history_event_type.TOPICAGREEMENT_CONCLUDED
|
|
or event:getType() == df.history_event_type.TOPICAGREEMENT_REJECTED
|
|
or event:getType() == df.history_event_type.TOPICAGREEMENT_MADE
|
|
or event:getType() == df.history_event_type.BODY_ABUSED
|
|
or event:getType() == df.history_event_type.CHANGE_CREATURE_TYPE
|
|
or event:getType() == df.history_event_type.CHANGE_HF_JOB
|
|
or event:getType() == df.history_event_type.CHANGE_HF_STATE
|
|
or event:getType() == df.history_event_type.CREATED_BUILDING
|
|
or event:getType() == df.history_event_type.CREATURE_DEVOURED
|
|
or event:getType() == df.history_event_type.HF_DOES_INTERACTION
|
|
or event:getType() == df.history_event_type.HF_LEARNS_SECRET
|
|
or event:getType() == df.history_event_type.HIST_FIGURE_NEW_PET
|
|
or event:getType() == df.history_event_type.HIST_FIGURE_REACH_SUMMIT
|
|
or event:getType() == df.history_event_type.ITEM_STOLEN
|
|
or event:getType() == df.history_event_type.REMOVE_HF_ENTITY_LINK
|
|
or event:getType() == df.history_event_type.REMOVE_HF_SITE_LINK
|
|
or event:getType() == df.history_event_type.REPLACED_BUILDING
|
|
or event:getType() == df.history_event_type.MASTERPIECE_CREATED_ARCH_DESIGN
|
|
or event:getType() == df.history_event_type.MASTERPIECE_CREATED_DYE_ITEM
|
|
or event:getType() == df.history_event_type.MASTERPIECE_CREATED_ARCH_CONSTRUCT
|
|
or event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM
|
|
or event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM_IMPROVEMENT
|
|
or event:getType() == df.history_event_type.MASTERPIECE_CREATED_FOOD -- Missing item subtype
|
|
or event:getType() == df.history_event_type.MASTERPIECE_CREATED_ENGRAVING
|
|
or event:getType() == df.history_event_type.MASTERPIECE_LOST
|
|
or event:getType() == df.history_event_type.ENTITY_ACTION
|
|
or event:getType() == df.history_event_type.HF_ACT_ON_BUILDING
|
|
or event:getType() == df.history_event_type.ARTIFACT_CREATED
|
|
or event:getType() == df.history_event_type.ASSUME_IDENTITY
|
|
or event:getType() == df.history_event_type.CREATE_ENTITY_POSITION
|
|
or event:getType() == df.history_event_type.DIPLOMAT_LOST
|
|
or event:getType() == df.history_event_type.MERCHANT
|
|
or event:getType() == df.history_event_type.WAR_PEACE_ACCEPTED
|
|
or event:getType() == df.history_event_type.WAR_PEACE_REJECTED
|
|
or event:getType() == df.history_event_type.HIST_FIGURE_WOUNDED
|
|
or event:getType() == df.history_event_type.HIST_FIGURE_DIED
|
|
then
|
|
file:write("\t<historical_event>\n")
|
|
file:write("\t\t<id>"..event.id.."</id>\n")
|
|
file:write("\t\t<type>"..tostring(df.history_event_type[event:getType()]):lower().."</type>\n")
|
|
for k,v in pairs(event) do
|
|
if k == "year" or k == "seconds" or k == "flags" or k == "id"
|
|
or (k == "region" and event:getType() ~= df.history_event_type.HF_DOES_INTERACTION)
|
|
or k == "region_pos" or k == "layer" or k == "feature_layer" or k == "subregion"
|
|
or k == "anon_1" or k == "anon_2" or k == "flags2" or k == "unk1" then
|
|
|
|
elseif event:getType() == df.history_event_type.ADD_HF_ENTITY_LINK and k == "link_type" then
|
|
file:write("\t\t<"..k..">"..df.histfig_entity_link_type[v]:lower().."</"..k..">\n")
|
|
elseif event:getType() == df.history_event_type.ADD_HF_ENTITY_LINK and k == "position_id" then
|
|
local entity = findEntity(event.civ)
|
|
if (entity ~= nil and event.civ > -1 and v > -1) then
|
|
for entitypositionsK, entityPositionsV in ipairs(entity.positions.own) do
|
|
if entityPositionsV.id == v then
|
|
file:write("\t\t<position>"..tostring(entityPositionsV.name[0]):lower().."</position>\n")
|
|
break
|
|
end
|
|
end
|
|
else
|
|
file:write("\t\t<position>-1</position>\n")
|
|
end
|
|
elseif event:getType() == df.history_event_type.CREATE_ENTITY_POSITION and k == "position" then
|
|
local entity = findEntity(event.site_civ)
|
|
if (entity ~= nil and v > -1) then
|
|
for entitypositionsK, entityPositionsV in ipairs(entity.positions.own) do
|
|
if entityPositionsV.id == v then
|
|
file:write("\t\t<position>"..tostring(entityPositionsV.name[0]):lower().."</position>\n")
|
|
break
|
|
end
|
|
end
|
|
else
|
|
file:write("\t\t<position>-1</position>\n")
|
|
end
|
|
elseif event:getType() == df.history_event_type.REMOVE_HF_ENTITY_LINK and k == "link_type" then
|
|
file:write("\t\t<"..k..">"..df.histfig_entity_link_type[v]:lower().."</"..k..">\n")
|
|
elseif event:getType() == df.history_event_type.REMOVE_HF_ENTITY_LINK and k == "position_id" then
|
|
local entity = findEntity(event.civ)
|
|
if (entity ~= nil and event.civ > -1 and v > -1) then
|
|
for entitypositionsK, entityPositionsV in ipairs(entity.positions.own) do
|
|
if entityPositionsV.id == v then
|
|
file:write("\t\t<position>"..tostring(entityPositionsV.name[0]):lower().."</position>\n")
|
|
break
|
|
end
|
|
end
|
|
else
|
|
file:write("\t\t<position>-1</position>\n")
|
|
end
|
|
elseif event:getType() == df.history_event_type.ADD_HF_HF_LINK and k == "type" then
|
|
file:write("\t\t<link_type>"..df.histfig_hf_link_type[v]:lower().."</link_type>\n")
|
|
elseif event:getType() == df.history_event_type.ADD_HF_SITE_LINK and k == "type" then
|
|
file:write("\t\t<link_type>"..df.histfig_site_link_type[v]:lower().."</link_type>\n")
|
|
elseif event:getType() == df.history_event_type.REMOVE_HF_SITE_LINK and k == "type" then
|
|
file:write("\t\t<link_type>"..df.histfig_site_link_type[v]:lower().."</link_type>\n")
|
|
elseif (event:getType() == df.history_event_type.ITEM_STOLEN or
|
|
event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM or
|
|
event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM_IMPROVEMENT
|
|
) and k == "item_type" then
|
|
file:write("\t\t<item_type>"..df.item_type[v]:lower().."</item_type>\n")
|
|
elseif (event:getType() == df.history_event_type.ITEM_STOLEN or
|
|
event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM or
|
|
event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM_IMPROVEMENT
|
|
) and k == "item_subtype" then
|
|
--if event.item_type > -1 and v > -1 then
|
|
file:write("\t\t<"..k..">"..getItemSubTypeName(event.item_type,v).."</"..k..">\n")
|
|
--end
|
|
elseif event:getType() == df.history_event_type.ITEM_STOLEN and k == "mattype" then
|
|
if (v > -1) then
|
|
if (dfhack.matinfo.decode(event.mattype, event.matindex) == nil) then
|
|
file:write("\t\t<mattype>"..event.mattype.."</mattype>\n")
|
|
file:write("\t\t<matindex>"..event.matindex.."</matindex>\n")
|
|
else
|
|
file:write("\t\t<mat>"..dfhack.matinfo.toString(dfhack.matinfo.decode(event.mattype, event.matindex)).."</mat>\n")
|
|
end
|
|
end
|
|
elseif (event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM or
|
|
event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM_IMPROVEMENT
|
|
) and k == "mat_type" then
|
|
if (v > -1) then
|
|
if (dfhack.matinfo.decode(event.mat_type, event.mat_index) == nil) then
|
|
file:write("\t\t<mat_type>"..event.mat_type.."</mat_type>\n")
|
|
file:write("\t\t<mat_index>"..event.mat_index.."</mat_index>\n")
|
|
else
|
|
file:write("\t\t<mat>"..dfhack.matinfo.toString(dfhack.matinfo.decode(event.mat_type, event.mat_index)).."</mat>\n")
|
|
end
|
|
end
|
|
elseif event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM_IMPROVEMENT and k == "imp_mat_type" then
|
|
if (v > -1) then
|
|
if (dfhack.matinfo.decode(event.imp_mat_type, event.imp_mat_index) == nil) then
|
|
file:write("\t\t<imp_mat_type>"..event.imp_mat_type.."</imp_mat_type>\n")
|
|
file:write("\t\t<imp_mat_index>"..event.imp_mat_index.."</imp_mat_index>\n")
|
|
else
|
|
file:write("\t\t<imp_mat>"..dfhack.matinfo.toString(dfhack.matinfo.decode(event.imp_mat_type, event.imp_mat_index)).."</imp_mat>\n")
|
|
end
|
|
end
|
|
|
|
elseif event:getType() == df.history_event_type.ITEM_STOLEN and k == "matindex" then
|
|
--skip
|
|
elseif event:getType() == df.history_event_type.ITEM_STOLEN and k == "item" and v == -1 then
|
|
--skip
|
|
elseif (event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM or
|
|
event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM_IMPROVEMENT
|
|
) and k == "mat_index" then
|
|
--skip
|
|
elseif event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM_IMPROVEMENT and k == "imp_mat_index" then
|
|
--skip
|
|
elseif (event:getType() == df.history_event_type.WAR_PEACE_ACCEPTED or
|
|
event:getType() == df.history_event_type.WAR_PEACE_REJECTED or
|
|
event:getType() == df.history_event_type.TOPICAGREEMENT_CONCLUDED or
|
|
event:getType() == df.history_event_type.TOPICAGREEMENT_REJECTED or
|
|
event:getType() == df.history_event_type.TOPICAGREEMENT_MADE
|
|
) and k == "topic" then
|
|
file:write("\t\t<topic>"..tostring(df.meeting_topic[v]):lower().."</topic>\n")
|
|
elseif event:getType() == df.history_event_type.MASTERPIECE_CREATED_ITEM_IMPROVEMENT and k == "improvement_type" then
|
|
file:write("\t\t<improvement_type>"..df.improvement_type[v]:lower().."</improvement_type>\n")
|
|
elseif ((event:getType() == df.history_event_type.HIST_FIGURE_REACH_SUMMIT and k == "group")
|
|
or (event:getType() == df.history_event_type.HIST_FIGURE_NEW_PET and k == "group")
|
|
or (event:getType() == df.history_event_type.BODY_ABUSED and k == "bodies")) then
|
|
for detailK,detailV in pairs(v) do
|
|
file:write("\t\t<"..k..">"..detailV.."</"..k..">\n")
|
|
end
|
|
elseif event:getType() == df.history_event_type.HIST_FIGURE_NEW_PET and k == "pets" then
|
|
for detailK,detailV in pairs(v) do
|
|
file:write("\t\t<"..k..">"..df.global.world.raws.creatures.all[detailV].name[0].."</"..k..">\n")
|
|
end
|
|
elseif event:getType() == df.history_event_type.BODY_ABUSED and (k == "props") then
|
|
file:write("\t\t<"..k.."_item_type>"..tostring(df.item_type[event.props.item.item_type]):lower().."</"..k.."_item_type>\n")
|
|
file:write("\t\t<"..k.."_item_subtype>"..getItemSubTypeName(event.props.item.item_type,event.props.item.item_subtype).."</"..k.."_item_subtype>\n")
|
|
if (event.props.item.mat_type > -1) then
|
|
if (dfhack.matinfo.decode(event.props.item.mat_type, event.props.item.mat_index) == nil) then
|
|
file:write("\t\t<props_item_mat_type>"..event.props.item.mat_type.."</props_item_mat_type>\n")
|
|
file:write("\t\t<props_item_mat_index>"..event.props.item.mat_index.."</props_item_mat_index>\n")
|
|
else
|
|
file:write("\t\t<props_item_mat>"..dfhack.matinfo.toString(dfhack.matinfo.decode(event.props.item.mat_type, event.props.item.mat_index)).."</props_item_mat>\n")
|
|
end
|
|
end
|
|
--file:write("\t\t<"..k.."_item_mat_type>"..tostring(event.props.item.mat_type).."</"..k.."_item_mat_index>\n")
|
|
--file:write("\t\t<"..k.."_item_mat_index>"..tostring(event.props.item.mat_index).."</"..k.."_item_mat_index>\n")
|
|
file:write("\t\t<"..k.."_pile_type>"..tostring(event.props.pile_type).."</"..k.."_pile_type>\n")
|
|
elseif event:getType() == df.history_event_type.ASSUME_IDENTITY and k == "identity" then
|
|
if (table.contains(df.global.world.identities.all,v)) then
|
|
if (df.global.world.identities.all[v].histfig_id == -1) then
|
|
local thisIdentity = df.global.world.identities.all[v]
|
|
file:write("\t\t<identity_name>"..thisIdentity.name.first_name.."</identity_name>\n")
|
|
file:write("\t\t<identity_race>"..(df.global.world.raws.creatures.all[thisIdentity.race].creature_id):lower().."</identity_race>\n")
|
|
file:write("\t\t<identity_caste>"..(df.global.world.raws.creatures.all[thisIdentity.race].caste[thisIdentity.caste].caste_id):lower().."</identity_caste>\n")
|
|
else
|
|
file:write("\t\t<identity_hf>"..df.global.world.identities.all[v].histfig_id.."</identity_hf>\n")
|
|
end
|
|
end
|
|
elseif event:getType() == df.history_event_type.MASTERPIECE_CREATED_ARCH_CONSTRUCT and k == "building_type" then
|
|
file:write("\t\t<building_type>"..df.building_type[v]:lower().."</building_type>\n")
|
|
elseif event:getType() == df.history_event_type.MASTERPIECE_CREATED_ARCH_CONSTRUCT and k == "building_subtype" then
|
|
if (df.building_type[event.building_type]:lower() == "furnace") then
|
|
file:write("\t\t<building_subtype>"..df.furnace_type[v]:lower().."</building_subtype>\n")
|
|
elseif v > -1 then
|
|
file:write("\t\t<building_subtype>"..tostring(v).."</building_subtype>\n")
|
|
end
|
|
elseif k == "race" then
|
|
if v > -1 then
|
|
file:write("\t\t<race>"..df.global.world.raws.creatures.all[v].name[0].."</race>\n")
|
|
end
|
|
elseif k == "caste" then
|
|
if v > -1 then
|
|
file:write("\t\t<caste>"..(df.global.world.raws.creatures.all[event.race].caste[v].caste_id):lower().."</caste>\n")
|
|
end
|
|
elseif k == "interaction" and event:getType() == df.history_event_type.HF_DOES_INTERACTION then
|
|
file:write("\t\t<interaction_action>"..df.global.world.raws.interactions[v].str[3].value.."</interaction_action>\n")
|
|
file:write("\t\t<interaction_string>"..df.global.world.raws.interactions[v].str[4].value.."</interaction_string>\n")
|
|
elseif k == "interaction" and event:getType() == df.history_event_type.HF_LEARNS_SECRET then
|
|
file:write("\t\t<secret_text>"..df.global.world.raws.interactions[v].str[2].value.."</secret_text>\n")
|
|
elseif event:getType() == df.history_event_type.HIST_FIGURE_DIED and k == "weapon" then
|
|
for detailK,detailV in pairs(v) do
|
|
if (detailK == "item") then
|
|
if detailV > -1 then
|
|
file:write("\t\t<"..detailK..">"..detailV.."</"..detailK..">\n")
|
|
local thisItem = df.item.find(detailV)
|
|
if (thisItem ~= nil) then
|
|
if (thisItem.flags.artifact == true) then
|
|
for refk,refv in pairs(thisItem.general_refs) do
|
|
if (refv:getType() == df.general_ref_type.IS_ARTIFACT) then
|
|
file:write("\t\t<artifact_id>"..refv.artifact_id.."</artifact_id>\n")
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
end
|
|
elseif (detailK == "item_type") then
|
|
if event.weapon.item > -1 then
|
|
file:write("\t\t<"..detailK..">"..tostring(df.item_type[detailV]):lower().."</"..detailK..">\n")
|
|
end
|
|
elseif (detailK == "item_subtype") then
|
|
if event.weapon.item > -1 and detailV > -1 then
|
|
file:write("\t\t<"..detailK..">"..getItemSubTypeName(event.weapon.item_type,detailV).."</"..detailK..">\n")
|
|
end
|
|
elseif (detailK == "mattype") then
|
|
if (detailV > -1) then
|
|
file:write("\t\t<mat>"..dfhack.matinfo.toString(dfhack.matinfo.decode(event.weapon.mattype, event.weapon.matindex)).."</mat>\n")
|
|
end
|
|
elseif (detailK == "matindex") then
|
|
|
|
elseif (detailK == "shooter_item") then
|
|
if detailV > -1 then
|
|
file:write("\t\t<"..detailK..">"..detailV.."</"..detailK..">\n")
|
|
local thisItem = df.item.find(detailV)
|
|
if thisItem ~= nil then
|
|
if (thisItem.flags.artifact == true) then
|
|
for refk,refv in pairs(thisItem.general_refs) do
|
|
if (refv:getType() == df.general_ref_type.IS_ARTIFACT) then
|
|
file:write("\t\t<shooter_artifact_id>"..refv.artifact_id.."</shooter_artifact_id>\n")
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
elseif (detailK == "shooter_item_type") then
|
|
if event.weapon.shooter_item > -1 then
|
|
file:write("\t\t<"..detailK..">"..tostring(df.item_type[detailV]):lower().."</"..detailK..">\n")
|
|
end
|
|
elseif (detailK == "shooter_item_subtype") then
|
|
if event.weapon.shooter_item > -1 and detailV > -1 then
|
|
file:write("\t\t<"..detailK..">"..getItemSubTypeName(event.weapon.shooter_item_type,detailV).."</"..detailK..">\n")
|
|
end
|
|
elseif (detailK == "shooter_mattype") then
|
|
if (detailV > -1) then
|
|
file:write("\t\t<shooter_mat>"..dfhack.matinfo.toString(dfhack.matinfo.decode(event.weapon.shooter_mattype, event.weapon.shooter_matindex)).."</shooter_mat>\n")
|
|
end
|
|
elseif (detailK == "shooter_matindex") then
|
|
--skip
|
|
elseif detailK == "slayer_race" or detailK == "slayer_caste" then
|
|
--skip
|
|
else
|
|
file:write("\t\t<"..detailK..">"..detailV.."</"..detailK..">\n")
|
|
end
|
|
end
|
|
elseif event:getType() == df.history_event_type.HIST_FIGURE_DIED and k == "death_cause" then
|
|
file:write("\t\t<"..k..">"..df.death_type[v]:lower().."</"..k..">\n")
|
|
elseif event:getType() == df.history_event_type.CHANGE_HF_JOB and (k == "new_job" or k == "old_job") then
|
|
file:write("\t\t<"..k..">"..df.profession[v]:lower().."</"..k..">\n")
|
|
elseif event:getType() == df.history_event_type.CHANGE_CREATURE_TYPE and (k == "old_race" or k == "new_race") and v >= 0 then
|
|
file:write("\t\t<"..k..">"..df.global.world.raws.creatures.all[v].name[0].."</"..k..">\n")
|
|
else
|
|
file:write("\t\t<"..k..">"..tostring(v).."</"..k..">\n")
|
|
end
|
|
end
|
|
file:write("\t</historical_event>\n")
|
|
end
|
|
end
|
|
file:write("</historical_events>\n")
|
|
file:write("<historical_event_collections>\n")
|
|
file:write("</historical_event_collections>\n")
|
|
file:write("<historical_eras>\n")
|
|
file:write("</historical_eras>\n")
|
|
file:write("</df_world>\n")
|
|
file:close()
|
|
end
|
|
|
|
-- export information and XML ('p, x')
|
|
function export_legends_info()
|
|
print(' Exporting: World map/gen info')
|
|
gui.simulateInput(vs, 'LEGENDS_EXPORT_MAP')
|
|
print(' Exporting: Legends xml')
|
|
gui.simulateInput(vs, 'LEGENDS_EXPORT_XML')
|
|
print(" Exporting: Extra legends_plus xml")
|
|
export_more_legends_xml()
|
|
end
|
|
|
|
--- presses 'd' for detailed maps
|
|
function wait_for_legends_vs()
|
|
local vs = dfhack.gui.getCurViewscreen()
|
|
if i <= #MAPS then
|
|
if df.viewscreen_legendsst:is_instance(vs.parent) then
|
|
vs = vs.parent
|
|
end
|
|
if df.viewscreen_legendsst:is_instance(vs) then
|
|
gui.simulateInput(vs, 'LEGENDS_EXPORT_DETAILED_MAP')
|
|
dfhack.timeout(10,'frames',wait_for_export_maps_vs)
|
|
else
|
|
dfhack.timeout(10,'frames',wait_for_legends_vs)
|
|
end
|
|
end
|
|
end
|
|
|
|
-- selects detailed map and export it
|
|
function wait_for_export_maps_vs()
|
|
local vs = dfhack.gui.getCurViewscreen()
|
|
if dfhack.gui.getCurFocus() == "export_graphical_map" then
|
|
vs.sel_idx = i-1
|
|
print(' Exporting: '..MAPS[i]..' map')
|
|
gui.simulateInput(vs, 'SELECT')
|
|
i = i + 1
|
|
dfhack.timeout(10,'frames',wait_for_legends_vs)
|
|
else
|
|
dfhack.timeout(10,'frames',wait_for_export_maps_vs)
|
|
end
|
|
end
|
|
|
|
-- export site maps
|
|
function export_site_maps()
|
|
local vs = dfhack.gui.getCurViewscreen()
|
|
if ((dfhack.gui.getCurFocus() ~= "legends" ) and (not table.contains(vs, "main_cursor"))) then -- Using open-legends
|
|
vs = vs.parent
|
|
end
|
|
print(' Exporting: All possible site maps')
|
|
vs.main_cursor = 1
|
|
gui.simulateInput(vs, 'SELECT')
|
|
for i=1, #vs.sites do
|
|
gui.simulateInput(vs, 'LEGENDS_EXPORT_MAP')
|
|
gui.simulateInput(vs, 'STANDARDSCROLL_DOWN')
|
|
end
|
|
gui.simulateInput(vs, 'LEAVESCREEN')
|
|
end
|
|
|
|
-- main()
|
|
if dfhack.gui.getCurFocus() == "legends" or dfhack.gui.getCurFocus() == "dfhack/lua/legends" then
|
|
-- either native legends mode, or using the open-legends.lua script
|
|
if args[1] == "all" then
|
|
export_legends_info()
|
|
export_site_maps()
|
|
wait_for_legends_vs()
|
|
elseif args[1] == "info" then
|
|
export_legends_info()
|
|
elseif args[1] == "custom" then
|
|
export_more_legends_xml()
|
|
elseif args[1] == "maps" then
|
|
wait_for_legends_vs()
|
|
elseif args[1] == "sites" then
|
|
export_site_maps()
|
|
else dfhack.printerr('Valid arguments are "all", "info", "maps" or "sites"')
|
|
end
|
|
elseif args[1] == "maps" and
|
|
dfhack.gui.getCurFocus() == "export_graphical_map" then
|
|
wait_for_export_maps_vs()
|
|
else
|
|
dfhack.printerr('Exportlegends must be run from the main legends view')
|
|
end
|