dfhack/plugins/channel-safely/channel-manager.cpp

129 lines
4.9 KiB
C++

2022-11-06 01:12:35 -06:00
#include <channel-manager.h>
2022-11-06 16:59:30 -07:00
#include <tile-cache.h>
2022-11-06 01:12:35 -06:00
#include <inlines.h>
#include <modules/EventManager.h> //hash function for df::coord
2022-11-06 01:12:35 -06:00
#include <df/block_square_event_designation_priorityst.h>
#include <algorithm>
#include <random>
df::unit* find_dwarf(const df::coord &map_pos) {
df::unit* nearest = nullptr;
uint32_t distance;
for (auto unit : df::global::world->units.active) {
if (!nearest) {
nearest = unit;
distance = calc_distance(unit->pos, map_pos);
} else if (unit->status.labors[df::unit_labor::MINE]) {
uint32_t d = calc_distance(unit->pos, map_pos);
if (d < distance) {
nearest = unit;
distance = d;
} else if (Maps::canWalkBetween(unit->pos, map_pos)) {
return unit;
}
}
}
return nearest;
}
2022-11-06 01:12:35 -06:00
// sets mark flags as necessary, for all designations
2022-11-06 12:53:46 -07:00
void ChannelManager::manage_groups() {
INFO(manager).print("manage_groups()\n");
2022-11-06 01:12:35 -06:00
// make sure we've got a fort map to analyze
if (World::isFortressMode() && Maps::IsValid()) {
// iterate the groups we built/updated
for (const auto &group: groups) {
manage_group(group, true, has_any_groups_above(groups, group));
}
}
}
void ChannelManager::manage_group(const df::coord &map_pos, bool set_marker_mode, bool marker_mode) {
INFO(manager).print("manage_group(" COORD ")\n ", COORDARGS(map_pos));
if (!groups.count(map_pos)) {
groups.scan_one(map_pos);
}
auto iter = groups.find(map_pos);
if (iter != groups.end()) {
manage_group(*iter, set_marker_mode, marker_mode);
}
INFO(manager).print("manage_group() is done\n");
}
void ChannelManager::manage_group(const Group &group, bool set_marker_mode, bool marker_mode) {
INFO(manager).print("manage_group()\n");
if (!set_marker_mode) {
marker_mode = has_any_groups_above(groups, group);
2022-11-06 01:12:35 -06:00
}
for (auto &designation: group) {
manage_one(designation, true, marker_mode);
2022-11-06 01:12:35 -06:00
}
INFO(manager).print("manage_group() is done\n");
}
bool ChannelManager::manage_one(const df::coord &map_pos, bool set_marker_mode, bool marker_mode) {
2022-11-06 01:12:35 -06:00
if (Maps::isValidTilePos(map_pos)) {
TRACE(manager).print("manage_one((" COORD "), %d, %d)\n", COORDARGS(map_pos), set_marker_mode, marker_mode);
2022-11-06 01:12:35 -06:00
df::map_block* block = Maps::getTileBlock(map_pos);
// we calculate the position inside the block*
df::coord local(map_pos);
local.x = local.x % 16;
local.y = local.y % 16;
df::tile_occupancy &tile_occupancy = block->occupancy[Coord(local)];
// ensure that we aren't on the top-most layers
if (map_pos.z < mapz - 3) {
// do we already know whether to set marker mode?
if (set_marker_mode) {
TRACE(manager).print(" -> setting marker mode\n");
2022-11-06 14:16:27 -07:00
// if enabling marker mode, just do it
if (marker_mode) {
goto markerMode;
2022-11-06 14:16:27 -07:00
}
// otherwise we check for some safety
2022-11-06 14:16:27 -07:00
if (!is_channel_designation(block->designation[Coord(local)]) || is_safe_to_dig_down(map_pos)) {
// not a channel designation, or it is safe to dig down if it is
2022-11-06 16:59:30 -07:00
if (!block->flags.bits.designated) {
block->flags.bits.designated = true;
}
// we need to cache the tile now that we're activating the designation
2022-11-06 16:59:30 -07:00
TileCache::Get().cache(map_pos, block->tiletype[Coord(local)]);
TRACE(manager).print("marker mode DISABLED\n");
tile_occupancy.bits.dig_marked = false;
} else {
// we want to activate the designation, that's not what we get
goto markerMode;
2022-11-06 14:16:27 -07:00
}
2022-11-06 01:12:35 -06:00
} else {
// not set_marker_mode
TRACE(manager).print(" if(has_groups_above())\n");
// check that the group has no incomplete groups directly above it
if (has_group_above(groups, map_pos) || !is_safe_to_dig_down(map_pos)) {
markerMode:
TRACE(manager).print("marker mode ENABLED\n");
tile_occupancy.bits.dig_marked = true;
if (jobs.count(map_pos)) {
cancel_job(map_pos);
2022-11-06 01:12:35 -06:00
}
}
}
} else {
// if we are though, it should be totally safe to dig
tile_occupancy.bits.dig_marked = false;
}
TRACE(manager).print(" <- manage_one() exits normally\n");
return true;
2022-11-06 01:12:35 -06:00
}
return false;
}
void ChannelManager::mark_done(const df::coord &map_pos) {
groups.remove(map_pos);
2022-11-06 16:59:30 -07:00
jobs.erase(map_pos);
CSP::dignow_queue.erase(map_pos);
2022-11-06 16:59:30 -07:00
TileCache::Get().uncache(map_pos);
2022-11-06 01:12:35 -06:00
}