|
|
@ -1,4 +1,4 @@
|
|
|
|
// Designate all matching plants for gathering/cutting
|
|
|
|
// (un)designate matching plants for gathering/cutting
|
|
|
|
|
|
|
|
|
|
|
|
#include "Core.h"
|
|
|
|
#include "Core.h"
|
|
|
|
#include "Console.h"
|
|
|
|
#include "Console.h"
|
|
|
@ -19,137 +19,144 @@ using std::string;
|
|
|
|
using std::vector;
|
|
|
|
using std::vector;
|
|
|
|
using std::set;
|
|
|
|
using std::set;
|
|
|
|
using namespace DFHack;
|
|
|
|
using namespace DFHack;
|
|
|
|
|
|
|
|
using namespace df::enums;
|
|
|
|
|
|
|
|
|
|
|
|
using df::global::world;
|
|
|
|
using df::global::world;
|
|
|
|
|
|
|
|
|
|
|
|
DFhackCExport command_result df_getplants (Core * c, vector <string> & parameters)
|
|
|
|
DFhackCExport command_result df_getplants (Core * c, vector <string> & parameters)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
string plantMatStr = "";
|
|
|
|
string plantMatStr = "";
|
|
|
|
set<int> plantIDs;
|
|
|
|
set<int> plantIDs;
|
|
|
|
set<string> plantNames;
|
|
|
|
set<string> plantNames;
|
|
|
|
bool deselect = false, exclude = false, treesonly = false, shrubsonly = false;
|
|
|
|
bool deselect = false, exclude = false, treesonly = false, shrubsonly = false;
|
|
|
|
|
|
|
|
|
|
|
|
int count = 0;
|
|
|
|
int count = 0;
|
|
|
|
for (size_t i = 0; i < parameters.size(); i++)
|
|
|
|
for (size_t i = 0; i < parameters.size(); i++)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if(parameters[i] == "help" || parameters[i] == "?")
|
|
|
|
if(parameters[i] == "help" || parameters[i] == "?")
|
|
|
|
{
|
|
|
|
{
|
|
|
|
c->con.print("Specify the types of trees to cut down and/or shrubs to gather by their plant names, separated by spaces.\n"
|
|
|
|
c->con.print("Specify the types of trees to cut down and/or shrubs to gather by their plant IDs, separated by spaces.\n"
|
|
|
|
"Options:\n"
|
|
|
|
"Options:\n"
|
|
|
|
"\t-t - Select trees only (exclude shrubs)\n"
|
|
|
|
"\t-t - Select trees only (exclude shrubs)\n"
|
|
|
|
"\t-s - Select shrubs only (exclude trees)\n"
|
|
|
|
"\t-s - Select shrubs only (exclude trees)\n"
|
|
|
|
"\t-c - Clear designations instead of setting them\n"
|
|
|
|
"\t-c - Clear designations instead of setting them\n"
|
|
|
|
"\t-x - Apply selected action to all plants except those specified\n"
|
|
|
|
"\t-x - Apply selected action to all plants except those specified\n"
|
|
|
|
"Specifying both -t and -s will have no effect.\n"
|
|
|
|
"Specifying both -t and -s will have no effect.\n"
|
|
|
|
"If no plant IDs are specified, all valid plant IDs will be listed.\n"
|
|
|
|
"If no plant IDs are specified, all valid plant IDs will be listed.\n"
|
|
|
|
);
|
|
|
|
);
|
|
|
|
return CR_OK;
|
|
|
|
return CR_WRONG_USAGE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(parameters[i] == "-t")
|
|
|
|
else if(parameters[i] == "-t")
|
|
|
|
treesonly = true;
|
|
|
|
treesonly = true;
|
|
|
|
else if(parameters[i] == "-s")
|
|
|
|
else if(parameters[i] == "-s")
|
|
|
|
shrubsonly = true;
|
|
|
|
shrubsonly = true;
|
|
|
|
else if(parameters[i] == "-c")
|
|
|
|
else if(parameters[i] == "-c")
|
|
|
|
deselect = true;
|
|
|
|
deselect = true;
|
|
|
|
else if(parameters[i] == "-x")
|
|
|
|
else if(parameters[i] == "-x")
|
|
|
|
exclude = true;
|
|
|
|
exclude = true;
|
|
|
|
else plantNames.insert(parameters[i]);
|
|
|
|
else
|
|
|
|
}
|
|
|
|
plantNames.insert(parameters[i]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (treesonly && shrubsonly)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
c->con.printerr("Cannot specify both -t and -s at the same time!\n");
|
|
|
|
|
|
|
|
return CR_WRONG_USAGE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
CoreSuspender suspend(c);
|
|
|
|
CoreSuspender suspend(c);
|
|
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < world->raws.plants.all.size(); i++)
|
|
|
|
for (int i = 0; i < world->raws.plants.all.size(); i++)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
df::plant_raw *plant = world->raws.plants.all[i];
|
|
|
|
df::plant_raw *plant = world->raws.plants.all[i];
|
|
|
|
if (plantNames.find(plant->id) != plantNames.end())
|
|
|
|
if (plantNames.find(plant->id) != plantNames.end())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
plantNames.erase(plant->id);
|
|
|
|
plantNames.erase(plant->id);
|
|
|
|
plantIDs.insert(i);
|
|
|
|
plantIDs.insert(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (plantNames.size() > 0)
|
|
|
|
if (plantNames.size() > 0)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
c->con.printerr("Invalid plant ID(s):");
|
|
|
|
c->con.printerr("Invalid plant ID(s):");
|
|
|
|
for (set<string>::const_iterator it = plantNames.begin(); it != plantNames.end(); it++)
|
|
|
|
for (set<string>::const_iterator it = plantNames.begin(); it != plantNames.end(); it++)
|
|
|
|
c->con.printerr(" %s", it->c_str());
|
|
|
|
c->con.printerr(" %s", it->c_str());
|
|
|
|
c->con.printerr("\n");
|
|
|
|
c->con.printerr("\n");
|
|
|
|
return CR_FAILURE;
|
|
|
|
return CR_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (plantIDs.size() == 0)
|
|
|
|
if (plantIDs.size() == 0)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
c->con.print("Valid plant IDs:\n");
|
|
|
|
c->con.print("Valid plant IDs:\n");
|
|
|
|
for (int i = 0; i < world->raws.plants.all.size(); i++)
|
|
|
|
for (int i = 0; i < world->raws.plants.all.size(); i++)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
df::plant_raw *plant = world->raws.plants.all[i];
|
|
|
|
df::plant_raw *plant = world->raws.plants.all[i];
|
|
|
|
if (plant->flags.is_set(df::plant_raw_flags::GRASS))
|
|
|
|
if (plant->flags.is_set(plant_raw_flags::GRASS))
|
|
|
|
continue;
|
|
|
|
continue;
|
|
|
|
c->con.print("* (%s) %s - %s\n", plant->flags.is_set(df::plant_raw_flags::TREE) ? "tree" : "shrub", plant->id.c_str(), plant->name.c_str());
|
|
|
|
c->con.print("* (%s) %s - %s\n", plant->flags.is_set(plant_raw_flags::TREE) ? "tree" : "shrub", plant->id.c_str(), plant->name.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return CR_OK;
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
count = 0;
|
|
|
|
count = 0;
|
|
|
|
for (int i = 0; i < world->map.map_blocks.size(); i++)
|
|
|
|
for (int i = 0; i < world->map.map_blocks.size(); i++)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
df::map_block *cur = world->map.map_blocks[i];
|
|
|
|
df::map_block *cur = world->map.map_blocks[i];
|
|
|
|
bool dirty = false;
|
|
|
|
bool dirty = false;
|
|
|
|
for (int j = 0; j < cur->plants.size(); j++)
|
|
|
|
for (int j = 0; j < cur->plants.size(); j++)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
const df_plant *plant = (df_plant *)cur->plants[i];
|
|
|
|
const df::plant *plant = cur->plants[i];
|
|
|
|
int x = plant->x % 16;
|
|
|
|
int x = plant->pos.x % 16;
|
|
|
|
int y = plant->y % 16;
|
|
|
|
int y = plant->pos.y % 16;
|
|
|
|
if (plantIDs.find(plant->material) != plantIDs.end())
|
|
|
|
if (plantIDs.find(plant->material) != plantIDs.end())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (exclude)
|
|
|
|
if (exclude)
|
|
|
|
continue;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (!exclude)
|
|
|
|
if (!exclude)
|
|
|
|
continue;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TileShape shape = tileShape(cur->tiletype[x][y]);
|
|
|
|
TileShape shape = tileShape(cur->tiletype[x][y]);
|
|
|
|
if (plant->is_shrub && (treesonly || shape != SHRUB_OK))
|
|
|
|
if (plant->flags.bits.is_shrub && (treesonly || shape != SHRUB_OK))
|
|
|
|
continue;
|
|
|
|
continue;
|
|
|
|
if (!plant->is_shrub && (shrubsonly || (shape != TREE_OK && shape != TREE_DEAD)))
|
|
|
|
if (!plant->flags.bits.is_shrub && (shrubsonly || (shape != TREE_OK && shape != TREE_DEAD)))
|
|
|
|
continue;
|
|
|
|
continue;
|
|
|
|
if (cur->designation[x][y].bits.hidden)
|
|
|
|
if (cur->designation[x][y].bits.hidden)
|
|
|
|
continue;
|
|
|
|
continue;
|
|
|
|
if (deselect && cur->designation[x][y].bits.dig != df::tile_dig_designation::No)
|
|
|
|
if (deselect && cur->designation[x][y].bits.dig == tile_dig_designation::Default)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
cur->designation[x][y].bits.dig = df::tile_dig_designation::No;
|
|
|
|
cur->designation[x][y].bits.dig = tile_dig_designation::No;
|
|
|
|
dirty = true;
|
|
|
|
dirty = true;
|
|
|
|
++count;
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!deselect && cur->designation[x][y].bits.dig != df::tile_dig_designation::Default)
|
|
|
|
if (!deselect && cur->designation[x][y].bits.dig == tile_dig_designation::No)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
cur->designation[x][y].bits.dig = df::tile_dig_designation::Default;
|
|
|
|
cur->designation[x][y].bits.dig = tile_dig_designation::Default;
|
|
|
|
dirty = true;
|
|
|
|
dirty = true;
|
|
|
|
++count;
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (dirty)
|
|
|
|
if (dirty)
|
|
|
|
cur->flags.set(df::block_flags::Designated);
|
|
|
|
cur->flags.set(block_flags::Designated);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (count)
|
|
|
|
if (count)
|
|
|
|
c->con.print("Updated %d plant designations.\n", count);
|
|
|
|
c->con.print("Updated %d plant designations.\n", count);
|
|
|
|
return CR_OK;
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DFhackCExport const char * plugin_name ( void )
|
|
|
|
DFhackCExport const char * plugin_name ( void )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return "getplants";
|
|
|
|
return "getplants";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DFhackCExport command_result plugin_init ( Core * c, vector <PluginCommand> &commands)
|
|
|
|
DFhackCExport command_result plugin_init ( Core * c, vector <PluginCommand> &commands)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
commands.clear();
|
|
|
|
commands.clear();
|
|
|
|
commands.push_back(PluginCommand("getplants", "Cut down all of the specified trees or gather all of the specified shrubs", df_getplants));
|
|
|
|
commands.push_back(PluginCommand("getplants", "Cut down all of the specified trees or gather all of the specified shrubs", df_getplants));
|
|
|
|
return CR_OK;
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DFhackCExport command_result plugin_shutdown ( Core * c )
|
|
|
|
DFhackCExport command_result plugin_shutdown ( Core * c )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return CR_OK;
|
|
|
|
return CR_OK;
|
|
|
|
}
|
|
|
|
}
|