// De-ramp. All ramps marked for removal are replaced with given tile (presently, normal floor). #include #include #include #include #include #include using namespace std; #include #include #include #include #include #include using namespace DFHack; DFhackCExport command_result df_deramp (Core * c, vector & parameters); DFhackCExport const char * plugin_name ( void ) { return "deramp"; } DFhackCExport command_result plugin_init ( Core * c, std::vector &commands) { commands.clear(); commands.push_back(PluginCommand("deramp", "De-ramp. All ramps marked for removal are replaced with floors.", df_deramp)); return CR_OK; } DFhackCExport command_result plugin_shutdown ( Core * c ) { return CR_OK; } DFhackCExport command_result df_deramp (Core * c, vector & parameters) { uint32_t x_max,y_max,z_max; uint32_t num_blocks = 0; uint32_t bytes_read = 0; DFHack::designations40d designations; DFHack::tiletypes40d tiles; DFHack::tiletypes40d tilesAbove; //DFHack::TileRow *ptile; int32_t oldT, newT; bool dirty= false; int count=0; int countbad=0; for(int i = 0; i < parameters.size();i++) { if(parameters[i] == "help" || parameters[i] == "?") { c->con.print("This command does two things:\n" "* If there are any ramps designated for removal, thyw will be instantly removed.\n" "* Any ramps that don't have their counterpart will be removed (fixes bugs with caveins)\n" ); return CR_OK; } } c->Suspend(); DFHack::Maps *Mapz = c->getMaps(); // init the map if (!Mapz->Start()) { c->con.printerr("Can't init map.\n"); c->Resume(); return CR_FAILURE; } Mapz->getSize(x_max,y_max,z_max); uint8_t zeroes [16][16] = {0}; // walk the map for (uint32_t x = 0; x< x_max;x++) { for (uint32_t y = 0; y< y_max;y++) { for (uint32_t z = 0; z< z_max;z++) { if (Mapz->getBlock(x,y,z)) { dirty= false; Mapz->ReadDesignations(x,y,z, &designations); Mapz->ReadTileTypes(x,y,z, &tiles); if (Mapz->getBlock(x,y,z+1)) { Mapz->ReadTileTypes(x,y,z+1, &tilesAbove); } else { memset(&tilesAbove,0,sizeof(tilesAbove)); } for (uint32_t ty=0;ty<16;++ty) { for (uint32_t tx=0;tx<16;++tx) { //Only the remove ramp designation (ignore channel designation, etc) oldT = tiles[tx][ty]; if ( DFHack::designation_default == designations[tx][ty].bits.dig && DFHack::RAMP==DFHack::tileShape(oldT)) { //Current tile is a ramp. //Set current tile, as accurately as can be expected newT = DFHack::findSimilarTileType(oldT,DFHack::FLOOR); //If no change, skip it (couldn't find a good tile type) if ( oldT == newT) continue; //Set new tile type, clear designation tiles[tx][ty] = newT; designations[tx][ty].bits.dig = DFHack::designation_no; //Check the tile above this one, in case a downward slope needs to be removed. if ( DFHack::RAMP_TOP == DFHack::tileShape(tilesAbove[tx][ty]) ) { tilesAbove[tx][ty] = 32; } dirty= true; ++count; } // ramp fixer else if(DFHack::RAMP!=DFHack::tileShape(oldT) && DFHack::RAMP_TOP == DFHack::tileShape(tilesAbove[tx][ty])) { tilesAbove[tx][ty] = 32; countbad++; dirty = true; } } } //If anything was changed, write it all. if (dirty) { Mapz->WriteDesignations(x,y,z, &designations); Mapz->WriteTileTypes(x,y,z, &tiles); if (Mapz->getBlock(x,y,z+1)) { Mapz->WriteTileTypes(x,y,z+1, &tilesAbove); } dirty = false; } } } } } c->Resume(); if(count) c->con.print("Found and changed %d tiles.\n",count); if(countbad) c->con.print("Fixed %d bad down ramps.\n",countbad); return CR_OK; }