mirror of
https://github.com/Kelsidavis/WoWee.git
synced 2026-05-09 02:23:52 +00:00
feat(editor): add --info-project-tree for multi-zone project overview
Project-level tree view: every zone with quick counts + bake/viewer
artifact status. --info-zone-tree drills into one zone; this gives
the bird's-eye view across the whole project:
wowee_editor --info-project-tree custom_zones
custom_zones/ (2 zones, 2 tiles, 1 creatures, 1 objects, 1 quests)
├─ Empty/ (tiles=1, creat=0, obj=0, quest=0)
│ ├─ name : Empty
│ ├─ mapName : Empty
│ ├─ artifacts : (none)
│ └─ status : empty (only terrain)
└─ Forest/ (tiles=1, creat=1, obj=1, quest=1)
├─ name : Forest
├─ mapName : Forest
├─ artifacts : .glb
└─ status : populated
Per-zone summary line shows counts; sub-tree shows display name,
map slug, which derived artifacts have been baked (.glb / .obj /
.stl / .html / ZONE.md), and a populated/empty status.
Project-header line aggregates totals across all zones for the
'how big is my project?' answer in one glance.
Pairs with --info-tilemap (spatial coverage) and --zone-stats
(quantitative aggregate) — three different lenses on the project:
spatial, quantitative, and structural.
This commit is contained in:
parent
2152b230c8
commit
d048beaa1e
1 changed files with 86 additions and 2 deletions
|
|
@ -642,6 +642,8 @@ static void printUsage(const char* argv0) {
|
||||||
std::printf(" One-shot validate + creature/object/quest counts and exit\n");
|
std::printf(" One-shot validate + creature/object/quest counts and exit\n");
|
||||||
std::printf(" --info-zone-tree <zoneDir>\n");
|
std::printf(" --info-zone-tree <zoneDir>\n");
|
||||||
std::printf(" Render a hierarchical tree view of a zone's contents (no --json)\n");
|
std::printf(" Render a hierarchical tree view of a zone's contents (no --json)\n");
|
||||||
|
std::printf(" --info-project-tree <projectDir>\n");
|
||||||
|
std::printf(" Tree view of every zone in a project with quick counts (no --json)\n");
|
||||||
std::printf(" --info-zone-bytes <zoneDir> [--json]\n");
|
std::printf(" --info-zone-bytes <zoneDir> [--json]\n");
|
||||||
std::printf(" Per-file size breakdown grouped by category, sorted largest-first\n");
|
std::printf(" Per-file size breakdown grouped by category, sorted largest-first\n");
|
||||||
std::printf(" --info-zone-extents <zoneDir> [--json]\n");
|
std::printf(" --info-zone-extents <zoneDir> [--json]\n");
|
||||||
|
|
@ -792,8 +794,8 @@ int main(int argc, char* argv[]) {
|
||||||
"--info-glb-tree",
|
"--info-glb-tree",
|
||||||
"--validate-jsondbc", "--check-glb-bounds", "--validate-stl",
|
"--validate-jsondbc", "--check-glb-bounds", "--validate-stl",
|
||||||
"--validate-png", "--validate-blp",
|
"--validate-png", "--validate-blp",
|
||||||
"--zone-summary", "--info-zone-tree", "--info-zone-bytes",
|
"--zone-summary", "--info-zone-tree", "--info-project-tree",
|
||||||
"--info-zone-extents",
|
"--info-zone-bytes", "--info-zone-extents",
|
||||||
"--export-zone-summary-md", "--export-quest-graph",
|
"--export-zone-summary-md", "--export-quest-graph",
|
||||||
"--export-zone-csv", "--export-zone-html", "--export-project-html",
|
"--export-zone-csv", "--export-zone-html", "--export-project-html",
|
||||||
"--export-project-md", "--export-zone-checksum",
|
"--export-project-md", "--export-zone-checksum",
|
||||||
|
|
@ -4427,6 +4429,88 @@ int main(int argc, char* argv[]) {
|
||||||
std::printf(" %s%s\n", branch(last), diskFiles[k].c_str());
|
std::printf(" %s%s\n", branch(last), diskFiles[k].c_str());
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
} else if (std::strcmp(argv[i], "--info-project-tree") == 0 && i + 1 < argc) {
|
||||||
|
// Project-level tree view: every zone with quick counts +
|
||||||
|
// bake/viewer status. --info-zone-tree drills into one zone;
|
||||||
|
// this gives the bird's-eye view across the whole project.
|
||||||
|
std::string projectDir = argv[++i];
|
||||||
|
namespace fs = std::filesystem;
|
||||||
|
if (!fs::exists(projectDir) || !fs::is_directory(projectDir)) {
|
||||||
|
std::fprintf(stderr,
|
||||||
|
"info-project-tree: %s is not a directory\n",
|
||||||
|
projectDir.c_str());
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
struct ZE {
|
||||||
|
std::string name, dir, mapName;
|
||||||
|
int tiles = 0, creatures = 0, objects = 0, quests = 0;
|
||||||
|
bool hasGlb = false, hasObj = false, hasStl = false;
|
||||||
|
bool hasHtml = false, hasZoneMd = false;
|
||||||
|
};
|
||||||
|
std::vector<ZE> zones;
|
||||||
|
for (const auto& entry : fs::directory_iterator(projectDir)) {
|
||||||
|
if (!entry.is_directory()) continue;
|
||||||
|
if (!fs::exists(entry.path() / "zone.json")) continue;
|
||||||
|
wowee::editor::ZoneManifest zm;
|
||||||
|
if (!zm.load((entry.path() / "zone.json").string())) continue;
|
||||||
|
ZE z;
|
||||||
|
z.name = zm.displayName.empty() ? zm.mapName : zm.displayName;
|
||||||
|
z.dir = entry.path().filename().string();
|
||||||
|
z.mapName = zm.mapName;
|
||||||
|
z.tiles = static_cast<int>(zm.tiles.size());
|
||||||
|
wowee::editor::NpcSpawner sp;
|
||||||
|
if (sp.loadFromFile((entry.path() / "creatures.json").string())) {
|
||||||
|
z.creatures = static_cast<int>(sp.spawnCount());
|
||||||
|
}
|
||||||
|
wowee::editor::ObjectPlacer op;
|
||||||
|
if (op.loadFromFile((entry.path() / "objects.json").string())) {
|
||||||
|
z.objects = static_cast<int>(op.getObjects().size());
|
||||||
|
}
|
||||||
|
wowee::editor::QuestEditor qe;
|
||||||
|
if (qe.loadFromFile((entry.path() / "quests.json").string())) {
|
||||||
|
z.quests = static_cast<int>(qe.questCount());
|
||||||
|
}
|
||||||
|
z.hasGlb = fs::exists(entry.path() / (zm.mapName + ".glb"));
|
||||||
|
z.hasObj = fs::exists(entry.path() / (zm.mapName + ".obj"));
|
||||||
|
z.hasStl = fs::exists(entry.path() / (zm.mapName + ".stl"));
|
||||||
|
z.hasHtml = fs::exists(entry.path() / (zm.mapName + ".html"));
|
||||||
|
z.hasZoneMd = fs::exists(entry.path() / "ZONE.md");
|
||||||
|
zones.push_back(std::move(z));
|
||||||
|
}
|
||||||
|
std::sort(zones.begin(), zones.end(),
|
||||||
|
[](const ZE& a, const ZE& b) { return a.name < b.name; });
|
||||||
|
int totalTiles = 0, totalCreat = 0, totalObj = 0, totalQuest = 0;
|
||||||
|
for (const auto& z : zones) {
|
||||||
|
totalTiles += z.tiles; totalCreat += z.creatures;
|
||||||
|
totalObj += z.objects; totalQuest += z.quests;
|
||||||
|
}
|
||||||
|
std::printf("%s/ (%zu zones, %d tiles, %d creatures, %d objects, %d quests)\n",
|
||||||
|
projectDir.c_str(), zones.size(),
|
||||||
|
totalTiles, totalCreat, totalObj, totalQuest);
|
||||||
|
for (size_t k = 0; k < zones.size(); ++k) {
|
||||||
|
bool lastZ = (k == zones.size() - 1);
|
||||||
|
const auto& z = zones[k];
|
||||||
|
const char* zBranch = lastZ ? "└─ " : "├─ ";
|
||||||
|
const char* zCont = lastZ ? " " : "│ ";
|
||||||
|
std::printf("%s%s/ (tiles=%d, creat=%d, obj=%d, quest=%d)\n",
|
||||||
|
zBranch, z.dir.c_str(),
|
||||||
|
z.tiles, z.creatures, z.objects, z.quests);
|
||||||
|
// Artifact status row — quick visual of what's been baked.
|
||||||
|
std::printf("%s├─ name : %s\n", zCont, z.name.c_str());
|
||||||
|
std::printf("%s├─ mapName : %s\n", zCont, z.mapName.c_str());
|
||||||
|
std::printf("%s├─ artifacts : %s%s%s%s%s%s\n", zCont,
|
||||||
|
z.hasGlb ? ".glb " : "",
|
||||||
|
z.hasObj ? ".obj " : "",
|
||||||
|
z.hasStl ? ".stl " : "",
|
||||||
|
z.hasHtml ? ".html " : "",
|
||||||
|
z.hasZoneMd ? "ZONE.md " : "",
|
||||||
|
(!z.hasGlb && !z.hasObj && !z.hasStl &&
|
||||||
|
!z.hasHtml && !z.hasZoneMd) ? "(none)" : "");
|
||||||
|
std::printf("%s└─ status : %s\n", zCont,
|
||||||
|
(z.creatures || z.objects || z.quests) ?
|
||||||
|
"populated" : "empty (only terrain)");
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
} else if (std::strcmp(argv[i], "--info-zone-bytes") == 0 && i + 1 < argc) {
|
} else if (std::strcmp(argv[i], "--info-zone-bytes") == 0 && i + 1 < argc) {
|
||||||
// Per-file size breakdown grouped by category, sorted by size
|
// Per-file size breakdown grouped by category, sorted by size
|
||||||
// descending. Useful for capacity planning ('which file is
|
// descending. Useful for capacity planning ('which file is
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue