2015-02-22 12:41:45 -07:00
|
|
|
#include "Core.h"
|
|
|
|
#include <Console.h>
|
2022-11-29 01:58:43 -07:00
|
|
|
#include <Debug.h>
|
2015-02-22 12:41:45 -07:00
|
|
|
#include <Export.h>
|
|
|
|
#include <PluginManager.h>
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include "modules/Units.h"
|
|
|
|
#include "modules/World.h"
|
|
|
|
|
|
|
|
// DF data structure definition headers
|
|
|
|
#include "DataDefs.h"
|
|
|
|
#include <df/ui.h>
|
|
|
|
#include <df/world.h>
|
|
|
|
#include <df/unit.h>
|
|
|
|
#include <df/unit_soul.h>
|
|
|
|
#include <df/unit_labor.h>
|
|
|
|
#include <df/unit_skill.h>
|
|
|
|
#include <df/job.h>
|
|
|
|
#include <df/building.h>
|
|
|
|
#include <df/workshop_type.h>
|
|
|
|
#include <df/unit_misc_trait.h>
|
|
|
|
#include <df/entity_position_responsibility.h>
|
|
|
|
#include <df/historical_figure.h>
|
|
|
|
#include <df/historical_entity.h>
|
|
|
|
#include <df/histfig_entity_link.h>
|
|
|
|
#include <df/histfig_entity_link_positionst.h>
|
|
|
|
#include <df/entity_position_assignment.h>
|
|
|
|
#include <df/entity_position.h>
|
|
|
|
#include <df/building_tradedepotst.h>
|
|
|
|
#include <df/building_stockpilest.h>
|
|
|
|
#include <df/items_other_id.h>
|
|
|
|
#include <df/ui.h>
|
|
|
|
#include <df/activity_info.h>
|
|
|
|
|
|
|
|
#include <MiscUtils.h>
|
|
|
|
|
|
|
|
#include "modules/MapCache.h"
|
|
|
|
#include "modules/Items.h"
|
|
|
|
#include "modules/Units.h"
|
|
|
|
|
2022-11-20 09:09:52 -07:00
|
|
|
#include "laborstatemap.h"
|
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
using namespace DFHack;
|
|
|
|
using namespace df::enums;
|
|
|
|
|
|
|
|
DFHACK_PLUGIN("autohauler");
|
|
|
|
REQUIRE_GLOBAL(ui);
|
|
|
|
REQUIRE_GLOBAL(world);
|
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
#define ARRAY_COUNT(array) (sizeof(array)/sizeof((array)[0]))
|
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
/*
|
|
|
|
* Autohauler module for dfhack
|
2015-02-25 05:26:54 -07:00
|
|
|
* Fork of autolabor, DFHack version 0.40.24-r2
|
2015-02-22 12:41:45 -07:00
|
|
|
*
|
|
|
|
* Rather than the all-of-the-above means of autolabor, autohauler will instead
|
|
|
|
* only manage hauling labors and leave skilled labors entirely to the user, who
|
2015-02-25 05:22:25 -07:00
|
|
|
* will probably use Dwarf Therapist to do so.
|
2015-02-22 12:41:45 -07:00
|
|
|
* Idle dwarves will be assigned the hauling labors; everyone else (including
|
|
|
|
* those currently hauling) will have the hauling labors removed. This is to
|
|
|
|
* encourage every dwarf to do their assigned skilled labors whenever possible,
|
|
|
|
* but resort to hauling when those jobs are not available. This also implies
|
|
|
|
* that the user will have a very tight skill assignment, with most skilled
|
|
|
|
* labors only being assigned to just one dwarf, no dwarf having more than two
|
|
|
|
* active skilled labors, and almost every non-military dwarf having at least
|
|
|
|
* one skilled labor assigned.
|
2015-03-04 04:59:29 -07:00
|
|
|
* Autohauler allows skills to be flagged as to prevent hauling labors from
|
|
|
|
* being assigned when the skill is present. By default this is the unused
|
|
|
|
* ALCHEMIST labor but can be changed by the user.
|
2015-02-25 05:22:25 -07:00
|
|
|
* It is noteworthy that, as stated in autolabor.cpp, "for almost all labors,
|
2015-02-22 12:41:45 -07:00
|
|
|
* once a dwarf begins a job it will finish that job even if the associated
|
|
|
|
* labor is removed." This is why we can remove hauling labors by default to try
|
|
|
|
* to force dwarves to do "real" jobs whenever they can.
|
|
|
|
* This is a standalone plugin. However, it would be wise to delete
|
|
|
|
* autolabor.plug.dll as this plugin is mutually exclusive with it.
|
|
|
|
*/
|
|
|
|
|
|
|
|
DFHACK_PLUGIN_IS_ENABLED(enable_autohauler);
|
|
|
|
|
2022-11-29 01:58:43 -07:00
|
|
|
namespace DFHack {
|
|
|
|
DBG_DECLARE(autohauler, cycle, DebugCategory::LINFO);
|
|
|
|
}
|
2015-02-25 05:22:25 -07:00
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
static std::vector<int> state_count(NUM_STATE);
|
|
|
|
|
2015-03-04 04:59:29 -07:00
|
|
|
const static int DEFAULT_FRAME_SKIP = 30;
|
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
static PersistentDataItem config;
|
2015-03-04 04:59:29 -07:00
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
command_result autohauler (color_ostream &out, std::vector <std::string> & parameters);
|
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
static int frame_skip;
|
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
static bool isOptionEnabled(unsigned flag)
|
|
|
|
{
|
|
|
|
return config.isValid() && (config.ival(0) & flag) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum ConfigFlags {
|
|
|
|
CF_ENABLED = 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void setOptionEnabled(ConfigFlags flag, bool on)
|
|
|
|
{
|
|
|
|
if (!config.isValid())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (on)
|
|
|
|
config.ival(0) |= flag;
|
|
|
|
else
|
|
|
|
config.ival(0) &= ~flag;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum labor_mode {
|
2015-02-25 05:22:25 -07:00
|
|
|
ALLOW,
|
2015-02-22 12:41:45 -07:00
|
|
|
HAULERS,
|
2015-02-25 05:22:25 -07:00
|
|
|
FORBID
|
2015-02-22 12:41:45 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct labor_info
|
|
|
|
{
|
|
|
|
PersistentDataItem config;
|
|
|
|
|
|
|
|
int active_dwarfs;
|
|
|
|
|
|
|
|
labor_mode mode() { return (labor_mode) config.ival(0); }
|
2015-06-13 12:08:42 -06:00
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
void set_mode(labor_mode mode) { config.ival(0) = mode; }
|
2022-11-20 10:57:52 -07:00
|
|
|
|
|
|
|
void set_config(PersistentDataItem a) { config = a; }
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
struct labor_default
|
|
|
|
{
|
|
|
|
labor_mode mode;
|
|
|
|
int active_dwarfs;
|
2015-02-22 12:41:45 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static std::vector<struct labor_info> labor_infos;
|
|
|
|
|
|
|
|
static const struct labor_default default_labor_infos[] = {
|
2015-02-25 05:22:25 -07:00
|
|
|
/* MINE */ {ALLOW, 0},
|
2015-02-22 12:41:45 -07:00
|
|
|
/* HAUL_STONE */ {HAULERS, 0},
|
|
|
|
/* HAUL_WOOD */ {HAULERS, 0},
|
|
|
|
/* HAUL_BODY */ {HAULERS, 0},
|
|
|
|
/* HAUL_FOOD */ {HAULERS, 0},
|
|
|
|
/* HAUL_REFUSE */ {HAULERS, 0},
|
|
|
|
/* HAUL_ITEM */ {HAULERS, 0},
|
|
|
|
/* HAUL_FURNITURE */ {HAULERS, 0},
|
|
|
|
/* HAUL_ANIMAL */ {HAULERS, 0},
|
|
|
|
/* CLEAN */ {HAULERS, 0},
|
2015-02-25 05:22:25 -07:00
|
|
|
/* CUTWOOD */ {ALLOW, 0},
|
|
|
|
/* CARPENTER */ {ALLOW, 0},
|
|
|
|
/* DETAIL */ {ALLOW, 0},
|
|
|
|
/* MASON */ {ALLOW, 0},
|
|
|
|
/* ARCHITECT */ {ALLOW, 0},
|
|
|
|
/* ANIMALTRAIN */ {ALLOW, 0},
|
|
|
|
/* ANIMALCARE */ {ALLOW, 0},
|
|
|
|
/* DIAGNOSE */ {ALLOW, 0},
|
|
|
|
/* SURGERY */ {ALLOW, 0},
|
|
|
|
/* BONE_SETTING */ {ALLOW, 0},
|
|
|
|
/* SUTURING */ {ALLOW, 0},
|
|
|
|
/* DRESSING_WOUNDS */ {ALLOW, 0},
|
2015-06-13 12:08:42 -06:00
|
|
|
/* FEED_WATER_CIVILIANS */ {HAULERS, 0}, // This could also be ALLOW
|
2015-02-22 12:41:45 -07:00
|
|
|
/* RECOVER_WOUNDED */ {HAULERS, 0},
|
2015-02-25 05:22:25 -07:00
|
|
|
/* BUTCHER */ {ALLOW, 0},
|
|
|
|
/* TRAPPER */ {ALLOW, 0},
|
|
|
|
/* DISSECT_VERMIN */ {ALLOW, 0},
|
|
|
|
/* LEATHER */ {ALLOW, 0},
|
|
|
|
/* TANNER */ {ALLOW, 0},
|
|
|
|
/* BREWER */ {ALLOW, 0},
|
|
|
|
/* ALCHEMIST */ {FORBID, 0},
|
|
|
|
/* SOAP_MAKER */ {ALLOW, 0},
|
|
|
|
/* WEAVER */ {ALLOW, 0},
|
|
|
|
/* CLOTHESMAKER */ {ALLOW, 0},
|
|
|
|
/* MILLER */ {ALLOW, 0},
|
|
|
|
/* PROCESS_PLANT */ {ALLOW, 0},
|
|
|
|
/* MAKE_CHEESE */ {ALLOW, 0},
|
|
|
|
/* MILK */ {ALLOW, 0},
|
|
|
|
/* COOK */ {ALLOW, 0},
|
|
|
|
/* PLANT */ {ALLOW, 0},
|
|
|
|
/* HERBALIST */ {ALLOW, 0},
|
|
|
|
/* FISH */ {ALLOW, 0},
|
|
|
|
/* CLEAN_FISH */ {ALLOW, 0},
|
|
|
|
/* DISSECT_FISH */ {ALLOW, 0},
|
|
|
|
/* HUNT */ {ALLOW, 0},
|
|
|
|
/* SMELT */ {ALLOW, 0},
|
|
|
|
/* FORGE_WEAPON */ {ALLOW, 0},
|
|
|
|
/* FORGE_ARMOR */ {ALLOW, 0},
|
|
|
|
/* FORGE_FURNITURE */ {ALLOW, 0},
|
|
|
|
/* METAL_CRAFT */ {ALLOW, 0},
|
|
|
|
/* CUT_GEM */ {ALLOW, 0},
|
|
|
|
/* ENCRUST_GEM */ {ALLOW, 0},
|
|
|
|
/* WOOD_CRAFT */ {ALLOW, 0},
|
|
|
|
/* STONE_CRAFT */ {ALLOW, 0},
|
|
|
|
/* BONE_CARVE */ {ALLOW, 0},
|
|
|
|
/* GLASSMAKER */ {ALLOW, 0},
|
|
|
|
/* EXTRACT_STRAND */ {ALLOW, 0},
|
|
|
|
/* SIEGECRAFT */ {ALLOW, 0},
|
|
|
|
/* SIEGEOPERATE */ {ALLOW, 0},
|
|
|
|
/* BOWYER */ {ALLOW, 0},
|
|
|
|
/* MECHANIC */ {ALLOW, 0},
|
|
|
|
/* POTASH_MAKING */ {ALLOW, 0},
|
|
|
|
/* LYE_MAKING */ {ALLOW, 0},
|
|
|
|
/* DYER */ {ALLOW, 0},
|
|
|
|
/* BURN_WOOD */ {ALLOW, 0},
|
|
|
|
/* OPERATE_PUMP */ {ALLOW, 0},
|
|
|
|
/* SHEARER */ {ALLOW, 0},
|
|
|
|
/* SPINNER */ {ALLOW, 0},
|
|
|
|
/* POTTERY */ {ALLOW, 0},
|
|
|
|
/* GLAZING */ {ALLOW, 0},
|
|
|
|
/* PRESSING */ {ALLOW, 0},
|
|
|
|
/* BEEKEEPING */ {ALLOW, 0},
|
|
|
|
/* WAX_WORKING */ {ALLOW, 0},
|
2015-02-22 12:41:45 -07:00
|
|
|
/* HANDLE_VEHICLES */ {HAULERS, 0},
|
|
|
|
/* HAUL_TRADE */ {HAULERS, 0},
|
|
|
|
/* PULL_LEVER */ {HAULERS, 0},
|
|
|
|
/* REMOVE_CONSTRUCTION */ {HAULERS, 0},
|
|
|
|
/* HAUL_WATER */ {HAULERS, 0},
|
2015-02-25 05:22:25 -07:00
|
|
|
/* GELD */ {ALLOW, 0},
|
2015-02-22 12:41:45 -07:00
|
|
|
/* BUILD_ROAD */ {HAULERS, 0},
|
2015-12-09 18:03:35 -07:00
|
|
|
/* BUILD_CONSTRUCTION */ {HAULERS, 0},
|
|
|
|
/* PAPERMAKING */ {ALLOW, 0},
|
|
|
|
/* BOOKBINDING */ {ALLOW, 0}
|
2015-02-22 12:41:45 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct dwarf_info_t
|
|
|
|
{
|
|
|
|
dwarf_state state;
|
2015-06-13 12:08:42 -06:00
|
|
|
|
2015-02-25 05:22:25 -07:00
|
|
|
bool haul_exempt;
|
2015-02-22 12:41:45 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static void cleanup_state()
|
|
|
|
{
|
|
|
|
enable_autohauler = false;
|
|
|
|
labor_infos.clear();
|
|
|
|
}
|
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
static void reset_labor(df::unit_labor labor)
|
|
|
|
{
|
|
|
|
labor_infos[labor].set_mode(default_labor_infos[labor].mode);
|
|
|
|
}
|
|
|
|
|
2021-02-06 15:14:08 -07:00
|
|
|
static void enable_alchemist(color_ostream &out)
|
|
|
|
{
|
|
|
|
if (!Units::setLaborValidity(unit_labor::ALCHEMIST, true))
|
|
|
|
{
|
|
|
|
// informational only; this is a non-fatal error
|
2021-02-24 22:41:13 -07:00
|
|
|
out.printerr("%s: Could not flag Alchemist as a valid skill; Alchemist will not"
|
|
|
|
" be settable from DF or DFHack labor management screens.\n", plugin_name);
|
2021-02-06 15:14:08 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void init_state(color_ostream &out)
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
|
|
|
config = World::GetPersistentData("autohauler/config");
|
2015-02-25 05:22:25 -07:00
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
if (config.isValid() && config.ival(0) == -1)
|
|
|
|
config.ival(0) = 0;
|
|
|
|
|
|
|
|
enable_autohauler = isOptionEnabled(CF_ENABLED);
|
|
|
|
|
|
|
|
if (!enable_autohauler)
|
|
|
|
return;
|
|
|
|
|
2015-03-04 04:59:29 -07:00
|
|
|
auto cfg_frameskip = World::GetPersistentData("autohauler/frameskip");
|
|
|
|
if (cfg_frameskip.isValid())
|
|
|
|
{
|
|
|
|
frame_skip = cfg_frameskip.ival(0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cfg_frameskip = World::AddPersistentData("autohauler/frameskip");
|
|
|
|
cfg_frameskip.ival(0) = DEFAULT_FRAME_SKIP;
|
|
|
|
frame_skip = cfg_frameskip.ival(0);
|
|
|
|
}
|
2022-11-20 10:57:52 -07:00
|
|
|
labor_infos.resize(ARRAY_COUNT(default_labor_infos));
|
2015-03-04 04:59:29 -07:00
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
std::vector<PersistentDataItem> items;
|
|
|
|
World::GetPersistentData(&items, "autohauler/labors/", true);
|
2015-02-25 05:22:25 -07:00
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
for (auto& p : items)
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
2022-11-20 10:57:52 -07:00
|
|
|
std::string key = p.key();
|
2015-02-22 12:41:45 -07:00
|
|
|
df::unit_labor labor = (df::unit_labor) atoi(key.substr(strlen("autohauler/labors/")).c_str());
|
2018-04-10 12:29:00 -06:00
|
|
|
if (labor >= 0 && size_t(labor) < labor_infos.size())
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
2022-11-20 10:57:52 -07:00
|
|
|
labor_infos[labor].set_config(p);
|
2015-02-22 12:41:45 -07:00
|
|
|
labor_infos[labor].active_dwarfs = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add default labors for those not in save
|
2018-04-06 00:18:15 -06:00
|
|
|
for (size_t i = 0; i < ARRAY_COUNT(default_labor_infos); i++) {
|
2015-02-22 12:41:45 -07:00
|
|
|
if (labor_infos[i].config.isValid())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
std::stringstream name;
|
|
|
|
name << "autohauler/labors/" << i;
|
|
|
|
|
|
|
|
labor_infos[i].set_config(World::AddPersistentData(name.str()));
|
|
|
|
|
|
|
|
labor_infos[i].active_dwarfs = 0;
|
|
|
|
reset_labor((df::unit_labor) i);
|
|
|
|
}
|
2015-02-25 05:22:25 -07:00
|
|
|
|
2021-02-06 15:14:08 -07:00
|
|
|
enable_alchemist(out);
|
2015-02-22 12:41:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void enable_plugin(color_ostream &out)
|
|
|
|
{
|
|
|
|
if (!config.isValid())
|
|
|
|
{
|
|
|
|
config = World::AddPersistentData("autohauler/config");
|
|
|
|
config.ival(0) = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
setOptionEnabled(CF_ENABLED, true);
|
|
|
|
enable_autohauler = true;
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "Enabling the plugin." << std::endl;
|
2015-02-25 05:22:25 -07:00
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
cleanup_state();
|
2021-02-06 15:14:08 -07:00
|
|
|
init_state(out);
|
2015-02-22 12:41:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
DFhackCExport command_result plugin_init ( color_ostream &out, std::vector <PluginCommand> &commands)
|
|
|
|
{
|
|
|
|
if(ARRAY_COUNT(default_labor_infos) != ENUM_LAST_ITEM(unit_labor) + 1)
|
|
|
|
{
|
|
|
|
out.printerr("autohauler: labor size mismatch\n");
|
|
|
|
return CR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
commands.push_back(PluginCommand(
|
2022-07-18 15:33:24 -06:00
|
|
|
"autohauler",
|
|
|
|
"Automatically manage hauling labors.",
|
|
|
|
autohauler));
|
2015-02-22 12:41:45 -07:00
|
|
|
|
2021-02-06 15:14:08 -07:00
|
|
|
init_state(out);
|
2015-02-22 12:41:45 -07:00
|
|
|
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
DFhackCExport command_result plugin_shutdown ( color_ostream &out )
|
|
|
|
{
|
|
|
|
cleanup_state();
|
|
|
|
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
DFhackCExport command_result plugin_onstatechange(color_ostream &out, state_change_event event)
|
|
|
|
{
|
|
|
|
switch (event) {
|
|
|
|
case SC_MAP_LOADED:
|
|
|
|
cleanup_state();
|
2021-02-06 15:14:08 -07:00
|
|
|
init_state(out);
|
2015-02-22 12:41:45 -07:00
|
|
|
break;
|
|
|
|
case SC_MAP_UNLOADED:
|
|
|
|
cleanup_state();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
DFhackCExport command_result plugin_onupdate ( color_ostream &out )
|
|
|
|
{
|
|
|
|
static int step_count = 0;
|
2022-11-20 11:05:14 -07:00
|
|
|
if(!world || !world->map.block_index || !enable_autohauler)
|
|
|
|
{
|
|
|
|
return CR_OK;
|
2022-11-20 10:57:52 -07:00
|
|
|
}
|
2015-02-25 05:22:25 -07:00
|
|
|
|
2022-11-20 11:05:14 -07:00
|
|
|
if (++step_count < frame_skip)
|
2022-11-20 10:57:52 -07:00
|
|
|
return CR_OK;
|
2015-02-22 12:41:45 -07:00
|
|
|
step_count = 0;
|
|
|
|
|
|
|
|
std::vector<df::unit *> dwarfs;
|
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
for (auto& cre : world->units.active)
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
|
|
|
if (Units::isCitizen(cre))
|
|
|
|
{
|
|
|
|
dwarfs.push_back(cre);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int n_dwarfs = dwarfs.size();
|
|
|
|
|
|
|
|
if (n_dwarfs == 0)
|
|
|
|
return CR_OK;
|
|
|
|
|
|
|
|
std::vector<dwarf_info_t> dwarf_info(n_dwarfs);
|
|
|
|
|
|
|
|
state_count.clear();
|
|
|
|
state_count.resize(NUM_STATE);
|
|
|
|
|
|
|
|
for (int dwarf = 0; dwarf < n_dwarfs; dwarf++)
|
|
|
|
{
|
2015-02-25 05:22:25 -07:00
|
|
|
/* Before determining how to handle employment status, handle
|
|
|
|
* hauling exemptions first */
|
|
|
|
|
2015-06-13 12:08:42 -06:00
|
|
|
// Default deny condition of on break for later else-if series
|
|
|
|
bool is_migrant = false;
|
|
|
|
|
2015-02-25 05:22:25 -07:00
|
|
|
// Scan every labor. If a labor that disallows hauling is present
|
|
|
|
// for the dwarf, the dwarf is hauling exempt
|
|
|
|
FOR_ENUM_ITEMS(unit_labor, labor)
|
|
|
|
{
|
|
|
|
if (!(labor == unit_labor::NONE))
|
|
|
|
{
|
|
|
|
bool test1 = labor_infos[labor].mode() == FORBID;
|
|
|
|
bool test2 = dwarfs[dwarf]->status.labors[labor];
|
|
|
|
|
|
|
|
if(test1 && test2) dwarf_info[dwarf].haul_exempt = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan a dwarf's miscellaneous traits for on break or migrant status.
|
|
|
|
// If either of these are present, disable hauling because we want them
|
2015-06-13 12:19:14 -06:00
|
|
|
// to try to find real jobs first
|
2022-11-20 10:57:52 -07:00
|
|
|
auto v = dwarfs[dwarf]->status.misc_traits;
|
|
|
|
auto test_migrant = [](df::unit_misc_trait* t) { return t->id == misc_trait_type::Migrant; };
|
|
|
|
is_migrant = std::find_if(v.begin(), v.end(), test_migrant ) != v.end();
|
2015-02-25 05:22:25 -07:00
|
|
|
|
|
|
|
/* Now determine a dwarf's employment status and decide whether
|
|
|
|
* to assign hauling */
|
|
|
|
|
|
|
|
// I don't think you can set the labors for babies and children, but let's
|
2015-02-22 12:41:45 -07:00
|
|
|
// ignore them anyway
|
|
|
|
if (Units::isBaby(dwarfs[dwarf]) || Units::isChild(dwarfs[dwarf]))
|
|
|
|
{
|
|
|
|
dwarf_info[dwarf].state = CHILD;
|
|
|
|
}
|
2015-02-25 05:22:25 -07:00
|
|
|
// Account for any hauling exemptions here
|
|
|
|
else if (dwarf_info[dwarf].haul_exempt)
|
|
|
|
{
|
2015-06-13 12:08:42 -06:00
|
|
|
dwarf_info[dwarf].state = BUSY;
|
2015-02-25 05:22:25 -07:00
|
|
|
}
|
2015-06-13 12:08:42 -06:00
|
|
|
// Account for the military
|
2015-02-22 12:41:45 -07:00
|
|
|
else if (ENUM_ATTR(profession, military, dwarfs[dwarf]->profession))
|
|
|
|
dwarf_info[dwarf].state = MILITARY;
|
2018-11-24 12:52:26 -07:00
|
|
|
// Account for incoming migrants
|
2015-06-13 12:08:42 -06:00
|
|
|
else if (is_migrant)
|
|
|
|
{
|
|
|
|
dwarf_info[dwarf].state = OTHER;
|
|
|
|
}
|
2015-02-22 12:41:45 -07:00
|
|
|
else if (dwarfs[dwarf]->job.current_job == NULL)
|
|
|
|
{
|
2015-03-04 04:59:29 -07:00
|
|
|
dwarf_info[dwarf].state = IDLE;
|
2015-02-22 12:41:45 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int job = dwarfs[dwarf]->job.current_job->job_type;
|
2018-04-06 00:18:15 -06:00
|
|
|
if (job >= 0 && size_t(job) < ARRAY_COUNT(dwarf_states))
|
2015-02-22 12:41:45 -07:00
|
|
|
dwarf_info[dwarf].state = dwarf_states[job];
|
|
|
|
else
|
|
|
|
{
|
2022-11-29 01:58:43 -07:00
|
|
|
WARN(cycle, out).print("Dwarf %i \"%s\" has unknown job %i\n", dwarf, dwarfs[dwarf]->name.first_name.c_str(), job);
|
|
|
|
dwarf_info[dwarf].state = OTHER;
|
2015-02-22 12:41:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
state_count[dwarf_info[dwarf].state]++;
|
2015-02-25 05:26:54 -07:00
|
|
|
|
2022-12-01 08:00:50 -07:00
|
|
|
TRACE(cycle, out).print("Dwarf %i \"%s\": state %s\n",
|
2022-11-29 01:58:43 -07:00
|
|
|
dwarf, dwarfs[dwarf]->name.first_name.c_str(), state_names[dwarf_info[dwarf].state]);
|
2015-02-22 12:41:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// This is a vector of all the labors
|
|
|
|
std::vector<df::unit_labor> labors;
|
|
|
|
|
|
|
|
// For every labor...
|
|
|
|
FOR_ENUM_ITEMS(unit_labor, labor)
|
|
|
|
{
|
|
|
|
// Ignore all nonexistent labors
|
|
|
|
if (labor == unit_labor::NONE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Set number of active dwarves for this job to zero
|
|
|
|
labor_infos[labor].active_dwarfs = 0;
|
|
|
|
|
|
|
|
// And add the labor to the aforementioned vector of labors
|
|
|
|
labors.push_back(labor);
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is a different algorithm than Autolabor. Instead, the intent is to
|
|
|
|
// have "real" jobs filled first, then if nothing is available the dwarf
|
|
|
|
// instead resorts to hauling.
|
2015-02-25 05:22:25 -07:00
|
|
|
|
2015-06-13 12:08:42 -06:00
|
|
|
// IDLE - Enable hauling
|
|
|
|
// BUSY - Disable hauling
|
|
|
|
// OTHER - Enable hauling
|
|
|
|
// MILITARY - Enable hauling
|
2015-02-22 12:41:45 -07:00
|
|
|
|
2015-06-13 12:08:42 -06:00
|
|
|
// There was no reason to put potential haulers in an array. All of them are
|
|
|
|
// covered in the following for loop.
|
2015-02-22 12:41:45 -07:00
|
|
|
|
2015-02-25 05:22:25 -07:00
|
|
|
FOR_ENUM_ITEMS(unit_labor, labor)
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
|
|
|
if (labor == unit_labor::NONE)
|
|
|
|
continue;
|
|
|
|
if (labor_infos[labor].mode() != HAULERS)
|
|
|
|
continue;
|
|
|
|
|
2018-04-06 00:18:15 -06:00
|
|
|
for(size_t dwarf = 0; dwarf < dwarfs.size(); dwarf++)
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
2015-10-30 16:41:09 -06:00
|
|
|
if (!Units::isValidLabor(dwarfs[dwarf], labor))
|
|
|
|
continue;
|
2015-02-25 05:22:25 -07:00
|
|
|
|
2015-06-13 12:08:42 -06:00
|
|
|
// Set hauling labors based on employment states
|
|
|
|
if(dwarf_info[dwarf].state == IDLE) {
|
2015-02-22 12:41:45 -07:00
|
|
|
dwarfs[dwarf]->status.labors[labor] = true;
|
|
|
|
}
|
2015-06-13 12:08:42 -06:00
|
|
|
else if(dwarf_info[dwarf].state == MILITARY) {
|
|
|
|
dwarfs[dwarf]->status.labors[labor] = true;
|
|
|
|
}
|
|
|
|
else if(dwarf_info[dwarf].state == OTHER) {
|
|
|
|
dwarfs[dwarf]->status.labors[labor] = true;
|
|
|
|
}
|
|
|
|
else if(dwarf_info[dwarf].state == BUSY) {
|
2015-02-22 12:41:45 -07:00
|
|
|
dwarfs[dwarf]->status.labors[labor] = false;
|
|
|
|
}
|
2015-03-04 04:59:29 -07:00
|
|
|
// If at the end of this the dwarf has the hauling labor, increment the
|
|
|
|
// counter
|
|
|
|
if(dwarfs[dwarf]->status.labors[labor])
|
|
|
|
{
|
|
|
|
labor_infos[labor].active_dwarfs++;
|
|
|
|
}
|
2015-06-13 12:08:42 -06:00
|
|
|
// CHILD ignored
|
2015-02-22 12:41:45 -07:00
|
|
|
}
|
2015-02-25 05:22:25 -07:00
|
|
|
}
|
2015-02-22 12:41:45 -07:00
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
void print_labor (df::unit_labor labor, color_ostream &out)
|
|
|
|
{
|
|
|
|
std::string labor_name = ENUM_KEY_STR(unit_labor, labor);
|
|
|
|
out << labor_name << ": ";
|
|
|
|
for (int i = 0; i < 20 - (int)labor_name.length(); i++)
|
|
|
|
out << ' ';
|
|
|
|
if (labor_infos[labor].mode() == ALLOW) out << "allow" << std::endl;
|
|
|
|
else if(labor_infos[labor].mode() == FORBID) out << "forbid" << std::endl;
|
|
|
|
else if(labor_infos[labor].mode() == HAULERS)
|
|
|
|
{
|
|
|
|
out << "haulers, currently " << labor_infos[labor].active_dwarfs << " dwarfs" << std::endl;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out << "Warning: Invalid labor mode!" << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
DFhackCExport command_result plugin_enable ( color_ostream &out, bool enable )
|
|
|
|
{
|
|
|
|
if (!Core::getInstance().isWorldLoaded()) {
|
|
|
|
out.printerr("World is not loaded: please load a game first.\n");
|
|
|
|
return CR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (enable && !enable_autohauler)
|
|
|
|
{
|
|
|
|
enable_plugin(out);
|
|
|
|
}
|
|
|
|
else if(!enable && enable_autohauler)
|
|
|
|
{
|
|
|
|
enable_autohauler = false;
|
|
|
|
setOptionEnabled(CF_ENABLED, false);
|
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "Autohauler is disabled." << std::endl;
|
2015-02-22 12:41:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
command_result autohauler (color_ostream &out, std::vector <std::string> & parameters)
|
|
|
|
{
|
|
|
|
CoreSuspender suspend;
|
|
|
|
|
|
|
|
if (!Core::getInstance().isWorldLoaded()) {
|
|
|
|
out.printerr("World is not loaded: please load a game first.\n");
|
|
|
|
return CR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parameters.size() == 1 &&
|
|
|
|
(parameters[0] == "0" || parameters[0] == "enable" ||
|
|
|
|
parameters[0] == "1" || parameters[0] == "disable"))
|
|
|
|
{
|
|
|
|
bool enable = (parameters[0] == "1" || parameters[0] == "enable");
|
|
|
|
|
|
|
|
return plugin_enable(out, enable);
|
|
|
|
}
|
2015-03-04 04:59:29 -07:00
|
|
|
else if (parameters.size() == 2 && parameters[0] == "frameskip")
|
|
|
|
{
|
|
|
|
auto cfg_frameskip = World::GetPersistentData("autohauler/frameskip");
|
|
|
|
if(cfg_frameskip.isValid())
|
|
|
|
{
|
|
|
|
int newValue = atoi(parameters[1].c_str());
|
|
|
|
cfg_frameskip.ival(0) = newValue;
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "Setting frame skip to " << newValue << std::endl;
|
2015-03-04 04:59:29 -07:00
|
|
|
frame_skip = cfg_frameskip.ival(0);
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "Warning! No persistent data for frame skip!" << std::endl;
|
2015-03-04 04:59:29 -07:00
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
}
|
2015-02-22 12:41:45 -07:00
|
|
|
else if (parameters.size() >= 2 && parameters.size() <= 4)
|
|
|
|
{
|
|
|
|
if (!enable_autohauler)
|
|
|
|
{
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "Error: The plugin is not enabled." << std::endl;
|
2015-02-22 12:41:45 -07:00
|
|
|
return CR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
df::unit_labor labor = unit_labor::NONE;
|
|
|
|
|
|
|
|
FOR_ENUM_ITEMS(unit_labor, test_labor)
|
|
|
|
{
|
|
|
|
if (parameters[0] == ENUM_KEY_STR(unit_labor, test_labor))
|
|
|
|
labor = test_labor;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (labor == unit_labor::NONE)
|
|
|
|
{
|
|
|
|
out.printerr("Could not find labor %s.\n", parameters[0].c_str());
|
|
|
|
return CR_WRONG_USAGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parameters[1] == "haulers")
|
|
|
|
{
|
|
|
|
labor_infos[labor].set_mode(HAULERS);
|
|
|
|
print_labor(labor, out);
|
|
|
|
return CR_OK;
|
|
|
|
}
|
2015-02-25 05:22:25 -07:00
|
|
|
if (parameters[1] == "allow")
|
|
|
|
{
|
|
|
|
labor_infos[labor].set_mode(ALLOW);
|
|
|
|
print_labor(labor, out);
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
if (parameters[1] == "forbid")
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
2015-02-25 05:22:25 -07:00
|
|
|
labor_infos[labor].set_mode(FORBID);
|
2015-02-22 12:41:45 -07:00
|
|
|
print_labor(labor, out);
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
if (parameters[1] == "reset")
|
|
|
|
{
|
|
|
|
reset_labor(labor);
|
|
|
|
print_labor(labor, out);
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
print_labor(labor, out);
|
|
|
|
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
else if (parameters.size() == 1 && parameters[0] == "reset-all")
|
|
|
|
{
|
|
|
|
if (!enable_autohauler)
|
|
|
|
{
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "Error: The plugin is not enabled." << std::endl;
|
2015-02-22 12:41:45 -07:00
|
|
|
return CR_FAILURE;
|
|
|
|
}
|
|
|
|
|
2018-04-06 00:18:15 -06:00
|
|
|
for (size_t i = 0; i < labor_infos.size(); i++)
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
|
|
|
reset_labor((df::unit_labor) i);
|
|
|
|
}
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "All labors reset." << std::endl;
|
2015-02-22 12:41:45 -07:00
|
|
|
return CR_OK;
|
|
|
|
}
|
2015-12-13 11:23:56 -07:00
|
|
|
else if (parameters.size() == 1 && (parameters[0] == "list" || parameters[0] == "status"))
|
2015-02-22 12:41:45 -07:00
|
|
|
{
|
|
|
|
if (!enable_autohauler)
|
|
|
|
{
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "Error: The plugin is not enabled." << std::endl;
|
2015-02-22 12:41:45 -07:00
|
|
|
return CR_FAILURE;
|
|
|
|
}
|
|
|
|
|
2015-03-04 04:59:29 -07:00
|
|
|
bool need_comma = false;
|
2015-02-22 12:41:45 -07:00
|
|
|
for (int i = 0; i < NUM_STATE; i++)
|
|
|
|
{
|
|
|
|
if (state_count[i] == 0)
|
|
|
|
continue;
|
|
|
|
if (need_comma)
|
|
|
|
out << ", ";
|
|
|
|
out << state_count[i] << ' ' << state_names[i];
|
2015-03-04 04:59:29 -07:00
|
|
|
need_comma = true;
|
2015-02-22 12:41:45 -07:00
|
|
|
}
|
2022-11-20 10:57:52 -07:00
|
|
|
out << std::endl;
|
2015-02-22 12:41:45 -07:00
|
|
|
|
2022-11-20 10:57:52 -07:00
|
|
|
out << "Autohauler is running every " << frame_skip << " frames." << std::endl;
|
2015-03-04 04:59:29 -07:00
|
|
|
|
2015-02-22 12:41:45 -07:00
|
|
|
if (parameters[0] == "list")
|
|
|
|
{
|
|
|
|
FOR_ENUM_ITEMS(unit_labor, labor)
|
|
|
|
{
|
|
|
|
if (labor == unit_labor::NONE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
print_labor(labor, out);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out.print("Automatically assigns hauling labors to dwarves.\n"
|
|
|
|
"Activate with 'enable autohauler', deactivate with 'disable autohauler'.\n"
|
|
|
|
"Current state: %d.\n", enable_autohauler);
|
|
|
|
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
}
|