|  |  | @ -52,7 +52,6 @@ using std::deque; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | DFHACK_PLUGIN("dwarfmonitor"); |  |  |  | DFHACK_PLUGIN("dwarfmonitor"); | 
			
		
	
		
		
			
				
					
					|  |  |  | DFHACK_PLUGIN_IS_ENABLED(is_enabled); |  |  |  | DFHACK_PLUGIN_IS_ENABLED(is_enabled); | 
			
		
	
		
		
			
				
					
					|  |  |  | REQUIRE_GLOBAL(current_weather); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | REQUIRE_GLOBAL(world); |  |  |  | REQUIRE_GLOBAL(world); | 
			
		
	
		
		
			
				
					
					|  |  |  | REQUIRE_GLOBAL(ui); |  |  |  | REQUIRE_GLOBAL(ui); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -74,20 +73,8 @@ struct less_second { | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  | }; |  |  |  | }; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | struct dwarfmonitor_configst { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     std::string date_format; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | }; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static dwarfmonitor_configst dwarfmonitor_config; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static bool monitor_jobs = false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static bool monitor_misery = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static bool monitor_date = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static bool monitor_weather = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static map<df::unit *, deque<activity_type>> work_history; |  |  |  | static map<df::unit *, deque<activity_type>> work_history; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | static int misery[] = { 0, 0, 0, 0, 0, 0, 0 }; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static bool misery_upto_date = false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static color_value monitor_colors[] = |  |  |  | static color_value monitor_colors[] = | 
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  | { | 
			
		
	
		
		
			
				
					
					|  |  |  |     COLOR_LIGHTRED, |  |  |  |     COLOR_LIGHTRED, | 
			
		
	
	
		
		
			
				
					|  |  | @ -151,102 +138,18 @@ static void move_cursor(df::coord &pos) | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | static void open_stats_screen(); |  |  |  | static void open_stats_screen(); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | namespace dm_lua { |  |  |  | static int getStressCategoryColors(lua_State *L) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     static color_ostream_proxy *out; |  |  |  |     const size_t n = sizeof(monitor_colors)/sizeof(color_value); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     static lua_State *state; |  |  |  |     lua_createtable(L, n, 0); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     typedef int(*initializer)(lua_State*); |  |  |  |     for (size_t i = 0; i < n; ++i) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     int no_args (lua_State *L) { return 0; } |  |  |  |         Lua::Push(L, monitor_colors[i]); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     void cleanup() |  |  |  |         lua_rawseti(L, -2, i+1); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (out) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             delete out; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             out = NULL; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     bool init_call (const char *func) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (!out) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             out = new color_ostream_proxy(Core::getInstance().getConsole()); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         return Lua::PushModulePublic(*out, state, "plugins.dwarfmonitor", func); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     bool safe_call (int nargs) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         return Lua::SafeCall(*out, state, nargs, 0); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     bool call (const char *func, initializer init = no_args) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         Lua::StackUnwinder top(state); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (!init_call(func)) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             return false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         int nargs = init(state); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         return safe_call(nargs); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     namespace api { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         int monitor_state (lua_State *L) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             std::string type = luaL_checkstring(L, 1); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             if (type == "weather") |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 lua_pushboolean(L, monitor_weather); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             else if (type == "misery") |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 lua_pushboolean(L, monitor_misery); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             else if (type == "date") |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 lua_pushboolean(L, monitor_date); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             else |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 lua_pushnil(L); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             return 1; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         int get_weather_counts (lua_State *L) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             #define WEATHER_TYPES WTYPE(clear, None); WTYPE(rain, Rain); WTYPE(snow, Snow); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             #define WTYPE(type, name) int type = 0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             WEATHER_TYPES |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             #undef WTYPE |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             int i, j; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             for (i = 0; i < 5; ++i) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 for (j = 0; j < 5; ++j) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                     switch ((*current_weather)[i][j]) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                         #define WTYPE(type, name) case weather_type::name: type++; break; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                         WEATHER_TYPES |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                         #undef WTYPE |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             lua_newtable(L); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             #define WTYPE(type, name) Lua::TableInsert(L, #type, type); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             WEATHER_TYPES |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             #undef WTYPE |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             #undef WEATHER_TYPES |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             return 1; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         int get_misery_data (lua_State *L) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             lua_newtable(L); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             for (int i = 0; i < 7; i++) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 Lua::Push(L, i); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 lua_newtable(L); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 Lua::TableInsert(L, "value", misery[i]); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 Lua::TableInsert(L, "color", monitor_colors[i]); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 Lua::TableInsert(L, "last", (i == 6)); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 lua_settable(L, -3); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             return 1; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     return 1; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | #define DM_LUA_FUNC(name) { #name, df::wrap_function(dm_lua::api::name, true) } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | #define DM_LUA_CMD(name) { #name, dm_lua::api::name } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | DFHACK_PLUGIN_LUA_COMMANDS { |  |  |  | DFHACK_PLUGIN_LUA_COMMANDS { | 
			
		
	
		
		
			
				
					
					|  |  |  |     DM_LUA_CMD(monitor_state), |  |  |  |     DFHACK_LUA_COMMAND(getStressCategoryColors), | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     DM_LUA_CMD(get_weather_counts), |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     DM_LUA_CMD(get_misery_data), |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     DFHACK_LUA_END |  |  |  |     DFHACK_LUA_END | 
			
		
	
		
		
			
				
					
					|  |  |  | }; |  |  |  | }; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -1648,8 +1551,7 @@ public: | 
			
		
	
		
		
			
				
					
					|  |  |  |         return (selected_column == 1) ? dwarf_column.getFirstSelectedElem() : nullptr; |  |  |  |         return (selected_column == 1) ? dwarf_column.getFirstSelectedElem() : nullptr; | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     void feed(set<df::interface_key> *input) |  |  |  |     void feed(set<df::interface_key> *input) override { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         bool key_processed = false; |  |  |  |         bool key_processed = false; | 
			
		
	
		
		
			
				
					
					|  |  |  |         switch (selected_column) |  |  |  |         switch (selected_column) | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |         { | 
			
		
	
	
		
		
			
				
					|  |  | @ -1723,8 +1625,7 @@ public: | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |         } | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     void render() |  |  |  |     void render() override { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         using namespace df::enums::interface_key; |  |  |  |         using namespace df::enums::interface_key; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (Screen::isDismissed(this)) |  |  |  |         if (Screen::isDismissed(this)) | 
			
		
	
	
		
		
			
				
					|  |  | @ -1751,7 +1652,7 @@ public: | 
			
		
	
		
		
			
				
					
					|  |  |  |             getSelectedUnit() ? COLOR_WHITE : COLOR_DARKGREY); |  |  |  |             getSelectedUnit() ? COLOR_WHITE : COLOR_DARKGREY); | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     std::string getFocusString() { return "dwarfmonitor_preferences"; } |  |  |  |     std::string getFocusString() override { return "dwarfmonitor_preferences"; } | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | private: |  |  |  | private: | 
			
		
	
		
		
			
				
					
					|  |  |  |     ListColumn<size_t> preferences_column; |  |  |  |     ListColumn<size_t> preferences_column; | 
			
		
	
	
		
		
			
				
					|  |  | @ -1762,13 +1663,11 @@ private: | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     vector<preference_map> preferences_store; |  |  |  |     vector<preference_map> preferences_store; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     void validateColumn() |  |  |  |     void validateColumn() { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         set_to_limit(selected_column, 1); |  |  |  |         set_to_limit(selected_column, 1); | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     void resize(int32_t x, int32_t y) |  |  |  |     void resize(int32_t x, int32_t y) override { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         dfhack_viewscreen::resize(x, y); |  |  |  |         dfhack_viewscreen::resize(x, y); | 
			
		
	
		
		
			
				
					
					|  |  |  |         preferences_column.resize(); |  |  |  |         preferences_column.resize(); | 
			
		
	
		
		
			
				
					
					|  |  |  |         dwarf_column.resize(); |  |  |  |         dwarf_column.resize(); | 
			
		
	
	
		
		
			
				
					|  |  | @ -1776,15 +1675,12 @@ private: | 
			
		
	
		
		
			
				
					
					|  |  |  | }; |  |  |  | }; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | static void open_stats_screen() |  |  |  | static void open_stats_screen() { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     Screen::show(dts::make_unique<ViewscreenFortStats>(), plugin_self); |  |  |  |     Screen::show(dts::make_unique<ViewscreenFortStats>(), plugin_self); | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | static void add_work_history(df::unit *unit, activity_type type) |  |  |  | static void add_work_history(df::unit *unit, activity_type type) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |     if (work_history.find(unit) == work_history.end()) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     if (work_history.find(unit) == work_history.end()) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         auto max_history = get_max_history(); |  |  |  |         auto max_history = get_max_history(); | 
			
		
	
		
		
			
				
					
					|  |  |  |         for (int i = 0; i < max_history; i++) |  |  |  |         for (int i = 0; i < max_history; i++) | 
			
		
	
		
		
			
				
					
					|  |  |  |             work_history[unit].push_back(JOB_UNKNOWN); |  |  |  |             work_history[unit].push_back(JOB_UNKNOWN); | 
			
		
	
	
		
		
			
				
					|  |  | @ -1794,8 +1690,7 @@ static void add_work_history(df::unit *unit, activity_type type) | 
			
		
	
		
		
			
				
					
					|  |  |  |     work_history[unit].pop_front(); |  |  |  |     work_history[unit].pop_front(); | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | static bool is_at_leisure(df::unit *unit) |  |  |  | static bool is_at_leisure(df::unit *unit) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     if (Units::getMiscTrait(unit, misc_trait_type::Migrant)) |  |  |  |     if (Units::getMiscTrait(unit, misc_trait_type::Migrant)) | 
			
		
	
		
		
			
				
					
					|  |  |  |         return true; |  |  |  |         return true; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -1805,32 +1700,17 @@ static bool is_at_leisure(df::unit *unit) | 
			
		
	
		
		
			
				
					
					|  |  |  |     return false; |  |  |  |     return false; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | static void reset() |  |  |  | static void reset() { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     work_history.clear(); |  |  |  |     work_history.clear(); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     for (int i = 0; i < 7; i++) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         misery[i] = 0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     misery_upto_date = false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | static void update_dwarf_stats(bool is_paused) |  |  |  | static void update_dwarf_stats(bool is_paused) | 
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  | { | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (monitor_misery) |  |  |  |     for (auto unit : world->units.active) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         for (int i = 0; i < 7; i++) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             misery[i] = 0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     for (auto iter = world->units.active.begin(); iter != world->units.active.end(); iter++) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         df::unit* unit = *iter; |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         if (!Units::isCitizen(unit)) |  |  |  |         if (!Units::isCitizen(unit)) | 
			
		
	
		
		
			
				
					
					|  |  |  |             continue; |  |  |  |             continue; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (!DFHack::Units::isActive(unit)) |  |  |  |         if (!DFHack::Units::isActive(unit)) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |             auto it = work_history.find(unit); |  |  |  |             auto it = work_history.find(unit); | 
			
		
	
		
		
			
				
					
					|  |  |  |             if (it != work_history.end()) |  |  |  |             if (it != work_history.end()) | 
			
		
	
		
		
			
				
					
					|  |  |  |                 work_history.erase(it); |  |  |  |                 work_history.erase(it); | 
			
		
	
	
		
		
			
				
					|  |  | @ -1838,35 +1718,25 @@ static void update_dwarf_stats(bool is_paused) | 
			
		
	
		
		
			
				
					
					|  |  |  |             continue; |  |  |  |             continue; | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |         } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (monitor_misery) |  |  |  |         if (is_paused) | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             misery[get_happiness_cat(unit)]++; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (!monitor_jobs || is_paused) |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |             continue; |  |  |  |             continue; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (Units::isBaby(unit) || |  |  |  |         if (Units::isBaby(unit) || | 
			
		
	
		
		
			
				
					
					|  |  |  |             Units::isChild(unit) || |  |  |  |                 Units::isChild(unit) || | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |             unit->profession == profession::DRUNK) |  |  |  |                 unit->profession == profession::DRUNK) | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |             continue; |  |  |  |             continue; | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (ENUM_ATTR(profession, military, unit->profession)) |  |  |  |         if (ENUM_ATTR(profession, military, unit->profession)) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |             add_work_history(unit, JOB_MILITARY); |  |  |  |             add_work_history(unit, JOB_MILITARY); | 
			
		
	
		
		
			
				
					
					|  |  |  |             continue; |  |  |  |             continue; | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |         } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (!unit->job.current_job) |  |  |  |         if (!unit->job.current_job) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |             add_work_history(unit, JOB_IDLE); |  |  |  |             add_work_history(unit, JOB_IDLE); | 
			
		
	
		
		
			
				
					
					|  |  |  |             continue; |  |  |  |             continue; | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |         } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (is_at_leisure(unit)) |  |  |  |         if (is_at_leisure(unit)) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |             add_work_history(unit, JOB_LEISURE); |  |  |  |             add_work_history(unit, JOB_LEISURE); | 
			
		
	
		
		
			
				
					
					|  |  |  |             continue; |  |  |  |             continue; | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |         } | 
			
		
	
	
		
		
			
				
					|  |  | @ -1876,107 +1746,21 @@ static void update_dwarf_stats(bool is_paused) | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | DFhackCExport command_result plugin_onupdate (color_ostream &out) |  |  |  | DFhackCExport command_result plugin_onupdate (color_ostream &out) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |     if (!is_enabled | !Maps::IsValid()) | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     if (!monitor_jobs && !monitor_misery) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         return CR_OK; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if(!Maps::IsValid()) |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         return CR_OK; |  |  |  |         return CR_OK; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     bool is_paused = DFHack::World::ReadPauseState(); |  |  |  |     bool is_paused = DFHack::World::ReadPauseState(); | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (is_paused) |  |  |  |     if (!is_paused && world->frame_counter % DELTA_TICKS != 0) | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |         return CR_OK; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         if (monitor_misery && !misery_upto_date) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             misery_upto_date = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         else |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             return CR_OK; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     else |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (world->frame_counter % DELTA_TICKS != 0) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             return CR_OK; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     update_dwarf_stats(is_paused); |  |  |  |     update_dwarf_stats(is_paused); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     return CR_OK; |  |  |  |     return CR_OK; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | struct dwarf_monitor_hook : public df::viewscreen_dwarfmodest |  |  |  | DFhackCExport command_result plugin_enable(color_ostream &, bool enable) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |     if (is_enabled != enable) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     typedef df::viewscreen_dwarfmodest interpose_base; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     DEFINE_VMETHOD_INTERPOSE(void, render, ()) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         INTERPOSE_NEXT(render)(); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         CoreSuspendClaimer suspend; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (Maps::IsValid()) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             dm_lua::call("render_all"); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | }; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | IMPLEMENT_VMETHOD_INTERPOSE(dwarf_monitor_hook, render); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static bool set_monitoring_mode(const string &mode, const bool &state) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     bool mode_recognized = false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (!is_enabled) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         return false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     /*
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         NOTE: although we are not touching DF directly but there might be |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         code running that uses these values. So this could use another mutex |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         or just suspend the core while we edit our values. |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     */ |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     CoreSuspender guard; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (mode == "work" || mode == "all") |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         mode_recognized = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         monitor_jobs = state; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (!monitor_jobs) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             reset(); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (mode == "misery" || mode == "all") |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         mode_recognized = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         monitor_misery = state; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (mode == "date" || mode == "all") |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         mode_recognized = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         monitor_date = state; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (mode == "weather" || mode == "all") |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         mode_recognized = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         monitor_weather = state; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     return mode_recognized; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static bool load_config() |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     return dm_lua::call("load_config"); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | DFhackCExport command_result plugin_enable(color_ostream &out, bool enable) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (enable) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         CoreSuspender guard; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         load_config(); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (is_enabled != enable) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (!INTERPOSE_HOOK(dwarf_monitor_hook, render).apply(enable)) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             return CR_FAILURE; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         reset(); |  |  |  |         reset(); | 
			
		
	
		
		
			
				
					
					|  |  |  |         is_enabled = enable; |  |  |  |         is_enabled = enable; | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
	
		
		
			
				
					|  |  | @ -1984,76 +1768,28 @@ DFhackCExport command_result plugin_enable(color_ostream &out, bool enable) | 
			
		
	
		
		
			
				
					
					|  |  |  |     return CR_OK; |  |  |  |     return CR_OK; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | static command_result dwarfmonitor_cmd(color_ostream &out, vector <string> & parameters) |  |  |  | static command_result dwarfmonitor_cmd(color_ostream &, vector <string> & parameters) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     bool show_help = false; |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     if (parameters.empty()) |  |  |  |     if (parameters.empty()) | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |         return CR_WRONG_USAGE; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         show_help = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     else |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         auto cmd = parameters[0][0]; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         string mode; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (parameters.size() > 1) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             mode = toLower(parameters[1]); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (cmd == 'v' || cmd == 'V') |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             out << "DwarfMonitor" << endl << "Version: " << PLUGIN_VERSION << endl; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         else if ((cmd == 'e' || cmd == 'E') && !mode.empty()) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             if (!is_enabled) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 plugin_enable(out, true); |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |             if (set_monitoring_mode(mode, true)) |  |  |  |     auto cmd = parameters[0][0]; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |             { |  |  |  |     if (cmd == 's' || cmd == 'S') { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |                 out << "Monitoring enabled: " << mode << endl; |  |  |  |         CoreSuspender guard; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |             } |  |  |  |         if(Maps::IsValid()) | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |             else |  |  |  |             Screen::show(dts::make_unique<ViewscreenFortStats>(), plugin_self); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |             { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 show_help = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         else if ((cmd == 'd' || cmd == 'D') && !mode.empty()) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             if (set_monitoring_mode(mode, false)) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 out << "Monitoring disabled: " << mode << endl; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             else |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 show_help = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         else if (cmd == 's' || cmd == 'S') |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             CoreSuspender guard; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             if(Maps::IsValid()) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 Screen::show(dts::make_unique<ViewscreenFortStats>(), plugin_self); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         else if (cmd == 'p' || cmd == 'P') |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             CoreSuspender guard; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             if(Maps::IsValid()) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |                 Screen::show(dts::make_unique<ViewscreenPreferences>(), plugin_self); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         else if (cmd == 'r' || cmd == 'R') |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             CoreSuspender guard; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             load_config(); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         else |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |             show_help = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |     else if (cmd == 'p' || cmd == 'P') { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     if (show_help) |  |  |  |         CoreSuspender guard; | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         if(Maps::IsValid()) | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |             Screen::show(dts::make_unique<ViewscreenPreferences>(), plugin_self); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     else | 
			
		
	
		
		
			
				
					
					|  |  |  |         return CR_WRONG_USAGE; |  |  |  |         return CR_WRONG_USAGE; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     return CR_OK; |  |  |  |     return CR_OK; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | DFhackCExport command_result plugin_init(color_ostream &out, std::vector <PluginCommand> &commands) |  |  |  | DFhackCExport command_result plugin_init(color_ostream &, std::vector <PluginCommand> &commands) | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  | { | 
			
		
	
		
		
			
				
					
					|  |  |  |     activity_labels[JOB_IDLE]               = "Idle"; |  |  |  |     activity_labels[JOB_IDLE]               = "Idle"; | 
			
		
	
		
		
			
				
					
					|  |  |  |     activity_labels[JOB_MILITARY]           = "Military Duty"; |  |  |  |     activity_labels[JOB_MILITARY]           = "Military Duty"; | 
			
		
	
	
		
		
			
				
					|  |  | @ -2079,27 +1815,13 @@ DFhackCExport command_result plugin_init(color_ostream &out, std::vector <Plugin | 
			
		
	
		
		
			
				
					
					|  |  |  |             "Measure fort happiness and efficiency.", |  |  |  |             "Measure fort happiness and efficiency.", | 
			
		
	
		
		
			
				
					
					|  |  |  |             dwarfmonitor_cmd)); |  |  |  |             dwarfmonitor_cmd)); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     dm_lua::state=Lua::Core::State; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (dm_lua::state == NULL) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         return CR_FAILURE; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     return CR_OK; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | DFhackCExport command_result plugin_shutdown(color_ostream &out) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     dm_lua::cleanup(); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     return CR_OK; |  |  |  |     return CR_OK; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | DFhackCExport command_result plugin_onstatechange(color_ostream &out, state_change_event event) |  |  |  | DFhackCExport command_result plugin_onstatechange(color_ostream &, state_change_event event) | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | { |  |  |  | { | 
			
		
	
		
		
			
				
					
					|  |  |  |     switch (event) { |  |  |  |     if (event == SC_MAP_LOADED) | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     case SC_MAP_LOADED: |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         reset(); |  |  |  |         reset(); | 
			
		
	
		
		
			
				
					
					|  |  |  |         break; |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     default: |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         break; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     return CR_OK; |  |  |  |     return CR_OK; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
	
		
		
			
				
					|  |  | 
 |