Skip to content

Commit f94b22a

Browse files
committedOct 24, 2022
Export additional editor data to vpps so it persists through import/export
Currently only exports object name + description
1 parent 1c751fd commit f94b22a

File tree

9 files changed

+149
-76
lines changed

9 files changed

+149
-76
lines changed
 

‎Nanoforge/gui/documents/TerritoryDocument.cpp

+13-73
Original file line numberDiff line numberDiff line change
@@ -577,27 +577,17 @@ void TerritoryDocument::DrawMenuBar(GuiState* state)
577577
}
578578

579579
bool exportPressed = false;
580-
bool exportPatchPressed = false;
581580
bool exportBinaryPatchPressed = false;
582581

583582
exportPressed = ImGui::Button("Export");
584-
if (!isSPMap)
585-
{
586-
exportPatchPressed = ImGui::Button("Export patch");
587-
ImGui::SameLine();
588-
gui::HelpMarker("This option is going away in future versions. You should use 'Export binary patch (SyncFaction)' instead.", ImGui::GetIO().FontDefault);
589-
exportBinaryPatchPressed = ImGui::Button("Export binary patch (SyncFaction)");
590-
}
591583

592584
MapExportType exportType;
593585
if (exportPressed)
594586
exportType = MapExportType::Vpp;
595-
else if (exportPatchPressed)
596-
exportType = MapExportType::RfgPatch;
597587
else if (exportBinaryPatchPressed)
598588
exportType = MapExportType::BinaryPatch;
599589

600-
if (exportPressed || exportPatchPressed || exportBinaryPatchPressed)
590+
if (exportPressed || exportBinaryPatchPressed)
601591
{
602592
exportTask_ = Task::Create("Export map");
603593
if (std::filesystem::exists(CVar_MapExportPath.Get<string>()))
@@ -1667,6 +1657,11 @@ void TerritoryDocument::Inspector(GuiState* state)
16671657
u32 handle = selectedObject_.Property("Handle").Get<u32>();
16681658
u32 num = selectedObject_.Property("Num").Get<u32>();
16691659
ImGui::Text(fmt::format("{}, {}", handle, num).c_str());
1660+
1661+
//Description
1662+
string description = selectedObject_.Get<string>("Description");
1663+
ImGui::InputText("Description", &description);
1664+
selectedObject_.Set<string>("Description", description);
16701665
ImGui::Separator();
16711666

16721667
if (ImGui::CollapsingHeader("Bounding box"))
@@ -2325,11 +2320,6 @@ void TerritoryDocument::ExportTask(GuiState* state, MapExportType exportType)
23252320
{
23262321
exportResult = ExportMapSP(state, CVar_MapExportPath.Get<string>());
23272322
}
2328-
else if (exportType == MapExportType::RfgPatch)
2329-
{
2330-
Log->error("Failed to export {}. RfgPatch export isn't supported for SP maps.", mapName);
2331-
exportResult = false;
2332-
}
23332323
else if (exportType == MapExportType::BinaryPatch)
23342324
{
23352325
Log->error("Failed to export {}. Binary patch export isn't supported for SP maps.", mapName);
@@ -2342,10 +2332,6 @@ void TerritoryDocument::ExportTask(GuiState* state, MapExportType exportType)
23422332
{
23432333
exportResult = ExportMap(state, CVar_MapExportPath.Get<string>());
23442334
}
2345-
else if (exportType == MapExportType::RfgPatch)
2346-
{
2347-
exportResult = ExportPatch();
2348-
}
23492335
else if (exportType == MapExportType::BinaryPatch)
23502336
{
23512337
exportResult = ExportBinaryPatch(state);
@@ -2380,7 +2366,7 @@ bool TerritoryDocument::ExportMap(GuiState* state, const string& exportPath)
23802366
string mapName = Path::GetFileNameNoExtension(Territory.Object.Get<string>("Name"));
23812367

23822368
//Steps: Export zones, extract vpp_pc, extract containers, copy zones to containers, repack containers, update asm_pc, repack vpp_pc, cleanup
2383-
const f32 numSteps = 8.0f;
2369+
const f32 numSteps = 9.0f;
23842370
const f32 percentagePerStep = 1.0f / numSteps;
23852371

23862372
exportPercentage_ = 0.0f;
@@ -2462,6 +2448,12 @@ bool TerritoryDocument::ExportMap(GuiState* state, const string& exportPath)
24622448
//Update asm_pc file
24632449
UpdateAsmPc(tempFolderPath + "\\vpp\\" + mapName + ".asm_pc");
24642450

2451+
exportPercentage_ += percentagePerStep;
2452+
exportStatus_ = "Exporting editor data...";
2453+
2454+
//Write additional editor data that isn't preserved in zone files
2455+
Exporters::ExportEditorMapData(Territory.Object, tempFolderPath + "vpp\\");
2456+
24652457
exportPercentage_ += percentagePerStep;
24662458
exportStatus_ = "Repacking " + terrPackfileName + "...";
24672459

@@ -2480,58 +2472,6 @@ bool TerritoryDocument::ExportMap(GuiState* state, const string& exportPath)
24802472
return true;
24812473
}
24822474

2483-
constexpr u32 RFG_PATCHFILE_SIGNATURE = 1330528590; //Equals ASCII string "NANO"
2484-
constexpr u32 RFG_PATCHFILE_VERSION = 1;
2485-
bool TerritoryDocument::ExportPatch()
2486-
{
2487-
exportPercentage_ = 0.0f;
2488-
2489-
//Export zone files
2490-
string terrPackfileName = Territory.Object.Get<string>("Name"); //Name of the map vpp_pc
2491-
string mapName = Path::GetFileNameNoExtension(terrPackfileName); //Name of the map
2492-
string exportFolderPath = CVar_MapExportPath.Get<string>(); //Folder to write final patch file to
2493-
if (!Exporters::ExportTerritory(Territory.Object, exportFolderPath))
2494-
{
2495-
LOG_ERROR("Map patch export failed! Map export failed for '{}'", mapName);
2496-
return false;
2497-
}
2498-
2499-
string zoneFilePath = fmt::format("{}\\{}.rfgzone_pc", exportFolderPath, mapName);
2500-
string pZoneFilePath = fmt::format("{}\\p_{}.rfgzone_pc", exportFolderPath, mapName);
2501-
std::vector<u8> zoneBytes = File::ReadAllBytes(zoneFilePath);
2502-
std::vector<u8> pZoneBytes = File::ReadAllBytes(pZoneFilePath);
2503-
if (zoneBytes.size() == 0 || pZoneBytes.size() == 0)
2504-
{
2505-
LOG_ERROR("Map patch export failed! More or both exported zone files are empty. Sizes: {}, {}", zoneBytes.size(), pZoneBytes.size());
2506-
return false;
2507-
}
2508-
2509-
//Merge zone files into single patch file. Meant to be used with the separate RfgMapPatcher tool. Makes the final exports much smaller.
2510-
//Write patch file header
2511-
BinaryWriter patch(fmt::format("{}\\{}.RfgPatch", exportFolderPath, mapName));
2512-
patch.Write<u32>(RFG_PATCHFILE_SIGNATURE);
2513-
patch.Write<u32>(RFG_PATCHFILE_VERSION);
2514-
patch.WriteNullTerminatedString(terrPackfileName);
2515-
patch.Align(4);
2516-
2517-
//Write zone file sizes
2518-
patch.Write<size_t>(zoneBytes.size());
2519-
patch.Write<size_t>(pZoneBytes.size());
2520-
2521-
//Write zone file data
2522-
patch.WriteSpan<u8>(zoneBytes);
2523-
patch.Align(4);
2524-
patch.WriteSpan<u8>(pZoneBytes);
2525-
patch.Align(4);
2526-
2527-
//Remove zone files. We only care about the final patch file with this option
2528-
std::filesystem::remove(zoneFilePath);
2529-
std::filesystem::remove(pZoneFilePath);
2530-
2531-
exportPercentage_ = 1.0f;
2532-
return true;
2533-
}
2534-
25352475
bool TerritoryDocument::ExportBinaryPatch(GuiState* state)
25362476
{
25372477
//Make sure folder exists for writing temporary files

‎Nanoforge/gui/documents/TerritoryDocument.h

-2
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,6 @@ class GuiState;
1414
enum MapExportType
1515
{
1616
Vpp,
17-
RfgPatch,
1817
BinaryPatch
1918
};
2019

@@ -79,7 +78,6 @@ class TerritoryDocument : public IDocument
7978

8079
void ExportTask(GuiState* state, MapExportType exportType);
8180
bool ExportMap(GuiState* state, const string& exportPath);
82-
bool ExportPatch();
8381
bool ExportBinaryPatch(GuiState* state);
8482
bool ExportMapSP(GuiState* state, const string& exportPath);
8583

‎Nanoforge/rfg/Territory.cpp

+12
Original file line numberDiff line numberDiff line change
@@ -133,6 +133,18 @@ void Territory::LoadThread(Handle<Task> task, Handle<Scene> scene, GuiState* sta
133133
}
134134
}
135135

136+
//Add description to objects missing them
137+
for (ObjectHandle zone : Object.GetObjectList("Zones"))
138+
{
139+
for (ObjectHandle obj : zone.GetObjectList("Objects"))
140+
{
141+
if (!obj.Has("Description"))
142+
{
143+
obj.Set<string>("Description", "");
144+
}
145+
}
146+
}
147+
136148
//Get zone name with most characters for UI purposes
137149
u32 longest = 0;
138150
for (ObjectHandle zone : Object.GetObjectList("Zones"))

‎Nanoforge/rfg/export/Exporters.h

+2
Original file line numberDiff line numberDiff line change
@@ -7,4 +7,6 @@ namespace Exporters
77
{
88
bool ExportTerritory(ObjectHandle territory, std::string_view outputPath, bool skipUneditedZones = false);
99
bool ExportZone(ObjectHandle zone, std::string_view outputPath, bool persistent = false);
10+
//Xml containing additional map data used by the editor which is stripped from the RFG zone files
11+
bool ExportEditorMapData(ObjectHandle territory, const string& outputFolder);
1012
}

‎Nanoforge/rfg/export/MapExporter.cpp

+29
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
#include "Exporters.h"
2+
#include <tinyxml2/tinyxml2.h>
23

34
bool Exporters::ExportTerritory(ObjectHandle territory, std::string_view outputPath, bool skipUneditedZones)
45
{
@@ -51,3 +52,31 @@ bool Exporters::ExportTerritory(ObjectHandle territory, std::string_view outputP
5152

5253
return true;
5354
}
55+
56+
bool Exporters::ExportEditorMapData(ObjectHandle territory, const string& outputFolder)
57+
{
58+
tinyxml2::XMLDocument doc;
59+
auto* editorData = doc.NewElement("EditorData");
60+
doc.InsertFirstChild(editorData);
61+
auto* zones = editorData->InsertNewChildElement("Zones");
62+
63+
//Store additional map data that typically isn't stored in RFG zone files (e.g. names, descriptions). This way it's preserved through map import/export
64+
for (ObjectHandle zone : territory.GetObjectList("Zones"))
65+
{
66+
auto* zoneXml = zones->InsertNewChildElement("Zone");
67+
zoneXml->InsertNewChildElement("Name")->SetText(zone.Get<string>("Name").c_str());
68+
auto* objectsXml = zoneXml->InsertNewChildElement("Objects");
69+
for (ObjectHandle object : zone.GetObjectList("Objects"))
70+
{
71+
auto* objectXml = objectsXml->InsertNewChildElement("Object");
72+
objectXml->InsertNewChildElement("Name")->SetText(object.Get<string>("Name").c_str());
73+
objectXml->InsertNewChildElement("Handle")->SetText(object.Get<u32>("Handle"));
74+
objectXml->InsertNewChildElement("Num")->SetText(object.Get<u32>("Num"));
75+
if (object.Has("Description"))
76+
objectXml->InsertNewChildElement("Description")->SetText(object.Get<string>("Description").c_str());
77+
}
78+
}
79+
80+
doc.SaveFile((outputFolder + "EditorData.xml").c_str());
81+
return true;
82+
}

‎Nanoforge/rfg/import/Importers.h

+3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
#pragma once
22
#include "application/Registry.h"
3+
#include <tinyxml2/tinyxml2.h>
34
#include <span>
45

56
class ZoneFile;
@@ -22,6 +23,8 @@ namespace Importers
2223
ObjectHandle ImportPegFromPath(std::string_view pegPath, TextureIndex* textureIndex, bool useExistingTextures = true);
2324
//Imports rfg chunk file (destructible mesh). Has extension .cchk_pc
2425
ObjectHandle ImportChunk(std::string_view chunkFilename, PackfileVFS* packfileVFS, TextureIndex* textureIndex, bool* stopSignal = nullptr, ObjectHandle textureCache = NullObjectHandle);
26+
//Xml containing additional map data used by the editor which is stripped from the RFG zone files
27+
bool ImportEditorMapData(ObjectHandle territory, tinyxml2::XMLDocument* doc);
2528
}
2629

2730
//Helpers for dealing with registry objects

‎Nanoforge/rfg/import/MapImporter.cpp

+88
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
#include "util/TaskScheduler.h"
99
#include "Log.h"
1010
#include <RfgTools++/formats/packfiles/Packfile3.h>
11+
#include <tinyxml2/tinyxml2.h>
1112

1213
//Used to stop the import process early by the calling code setting stopSignal to true. Does nothing if stopSignal is nullptr
1314
#define EarlyStopCheck() if(stopSignal && *stopSignal) \
@@ -106,6 +107,17 @@ ObjectHandle Importers::ImportTerritory(std::string_view territoryFilename, Pack
106107
}
107108
}
108109

110+
//Load additional map data if present
111+
if (packfile->Contains("EditorData.xml"))
112+
{
113+
auto editorDataBytes = packfile->ExtractSingleFile("EditorData.xml", true);
114+
std::vector<u8>& bytes = editorDataBytes.value();
115+
116+
tinyxml2::XMLDocument doc;
117+
doc.Parse((char*)bytes.data(), bytes.size());
118+
ImportEditorMapData(territory, &doc);
119+
}
120+
109121
return territory;
110122
}
111123

@@ -210,4 +222,80 @@ void LoadChunks(ObjectHandle zone, ObjectHandle territory, PackfileVFS* packfile
210222
obj.Set<ObjectHandle>("ChunkMesh", chunkMesh);
211223
}
212224
}
225+
}
226+
227+
bool Importers::ImportEditorMapData(ObjectHandle territory, tinyxml2::XMLDocument* doc)
228+
{
229+
auto* editorData = doc->FirstChildElement("EditorData");
230+
if (!editorData)
231+
{
232+
LOG_ERROR("<EditorData> not found in map editor data file.");
233+
return false;
234+
}
235+
236+
auto* zonesXml = editorData->FirstChildElement("Zones");
237+
if (!zonesXml)
238+
{
239+
LOG_ERROR("<EditorData/Zones> not found in map editor data file.");
240+
return false;
241+
}
242+
243+
//Load zone data
244+
for (auto* zoneXml = zonesXml->FirstChildElement("Zone"); zoneXml; zoneXml = zoneXml->NextSiblingElement())
245+
{
246+
const char* zoneName = zoneXml->FirstChildElement("Name")->GetText();
247+
if (!zoneName)
248+
{
249+
LOG_ERROR("<EditorData/Zones/Zone> missing <Name> in map editor data file.");
250+
continue;
251+
}
252+
253+
//Find zone
254+
auto search = std::ranges::find_if(territory.GetObjectList("Zones"), [&](ObjectHandle zone) -> bool { return zone.Get<string>("Name") == zoneName; });
255+
if (search == territory.GetObjectList("Zones").end())
256+
{
257+
LOG_ERROR("Failed to find zone with name '{}'", zoneName);
258+
continue;
259+
}
260+
ObjectHandle zone = *search;
261+
262+
//Load object data
263+
auto* objectsXml = zoneXml->FirstChildElement("Objects");
264+
for (auto* objectXml = objectsXml->FirstChildElement("Object"); objectXml; objectXml = objectXml->NextSiblingElement("Object"))
265+
{
266+
auto* handleXml = objectXml->FirstChildElement("Handle");
267+
auto* numXml = objectXml->FirstChildElement("Num");
268+
if (!handleXml || !handleXml->GetText())
269+
{
270+
LOG_ERROR("<Object/Handle> missing in map editor data file.");
271+
continue;
272+
}
273+
if (!numXml || !numXml->GetText())
274+
{
275+
LOG_ERROR("<Object/Num> missing in map editor data file.");
276+
continue;
277+
}
278+
279+
//Find object in registry
280+
u32 handle = handleXml->UnsignedText();
281+
u32 num = numXml->UnsignedText();
282+
auto objectSearch = std::ranges::find_if(zone.GetObjectList("Objects"), [&](ObjectHandle obj) -> bool { return obj.Get<u32>("Handle") == handle && obj.Get<u32>("Num") == num; });
283+
if (objectSearch == zone.GetObjectList("Objects").end())
284+
{
285+
LOG_ERROR("Failed to find zone object [{}, {}]", handle, num);
286+
continue;
287+
}
288+
ObjectHandle obj = *objectSearch;
289+
290+
//Set name + description if present
291+
auto* nameXml = objectXml->FirstChildElement("Name");
292+
auto* descriptionXml = objectXml->FirstChildElement("Description");
293+
if (nameXml && nameXml->GetText())
294+
obj.Set<string>("Name", nameXml->GetText());
295+
if (descriptionXml && descriptionXml->GetText())
296+
obj.Set<string>("Description", descriptionXml->GetText());
297+
}
298+
}
299+
300+
return true;
213301
}

‎Nanoforge/rfg/import/ZoneImporter.cpp

+1
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,7 @@ ObjectHandle Importers::ImportZoneFile(ZoneFile& zoneFile, ZoneFile& persistentZ
9292
string classname = current->Classname();
9393
ObjectHandle zoneObject = registry.CreateObject(classname);
9494
zone.AppendObjectList("Objects", zoneObject);
95+
zoneObject.Set<string>("Description", "");
9596
//TODO: Strip properties not needed by the editor
9697
zoneObject.Property("ClassnameHash").Set<u32>(current->ClassnameHash);
9798
zoneObject.Property("Handle").Set<u32>(current->Handle);

0 commit comments

Comments
 (0)
Please sign in to comment.