Veindig can now dig stairs to reach veins on different z-levels

develop
Petr Mrázek 2010-04-19 04:42:27 +02:00
parent 5b28ac3b59
commit 3268b26852
2 changed files with 127 additions and 41 deletions

@ -0,0 +1 @@
dfvdig.exe -x

@ -6,6 +6,7 @@
#include <stack> #include <stack>
#include <map> #include <map>
#include <stdio.h> #include <stdio.h>
#include <cstdlib>
using namespace std; using namespace std;
#include <DFTypes.h> #include <DFTypes.h>
@ -15,53 +16,66 @@ using namespace std;
#include <modules/Position.h> #include <modules/Position.h>
#include <modules/Materials.h> #include <modules/Materials.h>
#include <DFTileTypes.h> #include <DFTileTypes.h>
#include <argstream.h>
#define MAX_DIM 0x300 #define MAX_DIM 0x300
class Point class Point
{ {
public: public:
Point(uint32_t x, uint32_t y) Point(uint32_t x, uint32_t y, uint32_t z)
{ {
this->x = x; this->x = x;
this->y = y; this->y = y;
this->z = z;
}
Point()
{
x = y = z = 0;
} }
bool operator==(const Point &other) const bool operator==(const Point &other) const
{ {
return (other.x == x && other.y == y); return (other.x == x && other.y == y && other.z == z);
} }
bool operator<(const Point &other) const bool operator<(const Point &other) const
{ {
return ( (y*MAX_DIM + x) < (other.y*MAX_DIM + other.x)); return ( (y*z*1000 + y*MAX_DIM + x) < (other.y*other.z*1000 + other.y*MAX_DIM + other.x));
} }
Point operator/(int number) const Point operator/(int number) const
{ {
return Point(x/number, y/number); return Point(x/number, y/number, z);
} }
Point operator%(int number) const Point operator%(int number) const
{ {
return Point(x%number, y%number); return Point(x%number, y%number, z);
}
Point operator-(int number) const
{
return Point(x,y,z-number);
}
Point operator+(int number) const
{
return Point(x,y,z+number);
} }
uint32_t x; uint32_t x;
uint32_t y; uint32_t y;
uint32_t z;
}; };
class Block class Block
{ {
public: public:
Block(DFHack::Maps *_m, uint32_t x_, uint32_t y_, uint32_t z_) Block(DFHack::Maps *_m, Point _bcoord)
{ {
vector <DFHack::t_vein> veins; vector <DFHack::t_vein> veins;
m = _m; m = _m;
dirty = false; dirty = false;
valid = false; valid = false;
x = x_; bcoord = _bcoord;
y = y_; if(m->ReadBlock40d(bcoord.x,bcoord.y,bcoord.z,&raw))
z = z_;
if(m->ReadBlock40d(x_,y_,z_,&raw))
{ {
memset(materials,-1,sizeof(materials)); memset(materials,-1,sizeof(materials));
memset(bitmap,0,sizeof(bitmap)); memset(bitmap,0,sizeof(bitmap));
m->ReadVeins(x,y,z,&veins); m->ReadVeins(bcoord.x,bcoord.y,bcoord.z,&veins);
// for each vein // for each vein
for(int i = 0; i < (int)veins.size();i++) for(int i = 0; i < (int)veins.size();i++)
{ {
@ -119,8 +133,8 @@ class Block
if(dirty) if(dirty)
{ {
//printf("writing %d/%d/%d\n",x,y,z); //printf("writing %d/%d/%d\n",x,y,z);
m->WriteDesignations(x,y,z, &raw.designation); m->WriteDesignations(bcoord.x,bcoord.y,bcoord.z, &raw.designation);
m->WriteDirtyBit(x,y,z,true); m->WriteDirtyBit(bcoord.x,bcoord.y,bcoord.z,true);
} }
return true; return true;
} }
@ -128,27 +142,22 @@ class Block
volatile bool dirty; volatile bool dirty;
DFHack::Maps * m; DFHack::Maps * m;
DFHack::mapblock40d raw; DFHack::mapblock40d raw;
uint32_t x; Point bcoord;
uint32_t y;
uint32_t z;
int16_t materials[16][16]; int16_t materials[16][16];
int8_t bitmap[16][16]; int8_t bitmap[16][16];
}; };
class Layer class MapCache
{ {
public: public:
Layer(DFHack::Maps * Maps, uint32_t _z) MapCache(DFHack::Maps * Maps)
{ {
valid = 0; valid = 0;
this->Maps = Maps; this->Maps = Maps;
z = _z;
uint32_t z_max;
Maps->getSize(x_bmax, y_bmax, z_max); Maps->getSize(x_bmax, y_bmax, z_max);
if(z < z_max) valid = true;
valid = true;
}; };
~Layer() ~MapCache()
{ {
map<Point, Block *>::iterator p; map<Point, Block *>::iterator p;
for(p = blocks.begin(); p != blocks.end(); p++) for(p = blocks.begin(); p != blocks.end(); p++)
@ -173,9 +182,9 @@ class Layer
} }
else else
{ {
if(blockcoord.x < x_bmax && blockcoord.y < y_bmax) if(blockcoord.x < x_bmax && blockcoord.y < y_bmax && blockcoord.z < z_max)
{ {
Block * nblo = new Block(Maps,blockcoord.x,blockcoord.y,z); Block * nblo = new Block(Maps,blockcoord);
blocks[blockcoord] = nblo; blocks[blockcoord] = nblo;
return nblo; return nblo;
} }
@ -234,6 +243,15 @@ class Layer
} }
return false; return false;
} }
bool testCoord (Point tilecoord)
{
Block * b= BlockAt(tilecoord / 16);
if(b && b->valid)
{
return true;
}
return false;
}
bool WriteAll() bool WriteAll()
{ {
@ -247,17 +265,30 @@ class Layer
} }
private: private:
volatile bool valid; volatile bool valid;
uint32_t z;
uint32_t x_bmax; uint32_t x_bmax;
uint32_t y_bmax; uint32_t y_bmax;
uint32_t x_tmax; uint32_t x_tmax;
uint32_t y_tmax; uint32_t y_tmax;
uint32_t z_max;
DFHack::Maps * Maps; DFHack::Maps * Maps;
map<Point, Block *> blocks; map<Point, Block *> blocks;
}; };
int main (int argc, const char* argv[]) int main (int argc, char* argv[])
{ {
// Command line options
bool updown = false;
argstream as(argc,argv);
as >>option('x',"updown",updown,"Dig up and down stairs to reach other z-levels.")
>>help();
// sane check
if (!as.isOk())
{
cout << as.errorLog();
return 1;
}
DFHack::API DF("Memory.xml"); DFHack::API DF("Memory.xml");
try try
@ -303,7 +334,7 @@ int main (int argc, const char* argv[])
DF.Suspend(); DF.Suspend();
Pos->getCursorCoords(cx,cy,cz); Pos->getCursorCoords(cx,cy,cz);
} }
Point xy ((uint32_t)cx,(uint32_t)cy); Point xy ((uint32_t)cx,(uint32_t)cy,cz);
if(xy.x == 0 || xy.x == tx_max - 1 || xy.y == 0 || xy.y == ty_max - 1) if(xy.x == 0 || xy.x == tx_max - 1 || xy.y == 0 || xy.y == ty_max - 1)
{ {
cerr << "I won't dig the borders. That would be cheating!" << endl; cerr << "I won't dig the borders. That would be cheating!" << endl;
@ -313,7 +344,7 @@ int main (int argc, const char* argv[])
#endif #endif
return 1; return 1;
} }
Layer * L = new Layer(Maps,cz); MapCache * L = new MapCache(Maps);
DFHack::t_designation des = L->designationAt(xy); DFHack::t_designation des = L->designationAt(xy);
@ -345,39 +376,93 @@ int main (int argc, const char* argv[])
continue; continue;
// found a good tile, dig+unset material // found a good tile, dig+unset material
DFHack::t_designation des = L->designationAt(current); DFHack::t_designation des = L->designationAt(current);
des.bits.dig = DFHack::designation_default; DFHack::t_designation des_minus;
if(L->setDesignationAt(current,des)) DFHack::t_designation des_plus;
int16_t vmat_minus;
int16_t vmat_plus;
bool below = 0;
bool above = 0;
if(updown)
{
if(L->testCoord(current-1))
{
below = 1;
}
des_minus = L->designationAt(current-1);
vmat_minus = L->materialAt(current-1);
if(L->testCoord(current+1))
{
above = 1;
}
des_plus = L->designationAt(current+1);
vmat_plus = L->materialAt(current+1);
}
if(L->testCoord(current))
{ {
L->clearMaterialAt(current); L->clearMaterialAt(current);
if(current.x < tx_max - 2) if(current.x < tx_max - 2)
{ {
flood.push(Point(current.x + 1, current.y)); flood.push(Point(current.x + 1, current.y, current.z));
if(current.y < ty_max - 2) if(current.y < ty_max - 2)
{ {
flood.push(Point(current.x + 1, current.y + 1)); flood.push(Point(current.x + 1, current.y + 1,current.z));
flood.push(Point(current.x, current.y + 1)); flood.push(Point(current.x, current.y + 1,current.z));
} }
if(current.y > 1) if(current.y > 1)
{ {
flood.push(Point(current.x + 1, current.y - 1)); flood.push(Point(current.x + 1, current.y - 1,current.z));
flood.push(Point(current.x, current.y - 1)); flood.push(Point(current.x, current.y - 1,current.z));
} }
} }
if(current.x > 1) if(current.x > 1)
{ {
flood.push(Point(current.x - 1, current.y)); flood.push(Point(current.x - 1, current.y,current.z));
if(current.y < ty_max - 2) if(current.y < ty_max - 2)
{ {
flood.push(Point(current.x - 1, current.y + 1)); flood.push(Point(current.x - 1, current.y + 1,current.z));
flood.push(Point(current.x, current.y + 1)); flood.push(Point(current.x, current.y + 1,current.z));
} }
if(current.y > 1) if(current.y > 1)
{ {
flood.push(Point(current.x - 1, current.y - 1)); flood.push(Point(current.x - 1, current.y - 1,current.z));
flood.push(Point(current.x, current.y - 1)); flood.push(Point(current.x, current.y - 1,current.z));
}
}
if(updown)
{
if(current.z > 0 && below && vmat_minus == vmat2)
{
flood.push(Point(current.x, current.y,current.z - 1));
if(des_minus.bits.dig == DFHack::designation_d_stair)
des_minus.bits.dig = DFHack::designation_ud_stair;
else
des_minus.bits.dig = DFHack::designation_u_stair;
L->setDesignationAt(current-1,des_minus);
des.bits.dig = DFHack::designation_d_stair;
}
if(current.z < z_max - 1 && above && vmat_plus == vmat2)
{
flood.push(Point(current.x, current.y,current.z + 1));
if(des_plus.bits.dig == DFHack::designation_u_stair)
des_plus.bits.dig = DFHack::designation_ud_stair;
else
des_plus.bits.dig = DFHack::designation_d_stair;
L->setDesignationAt(current+1,des_plus);
if(des.bits.dig == DFHack::designation_d_stair)
des.bits.dig = DFHack::designation_ud_stair;
else
des.bits.dig = DFHack::designation_u_stair;
} }
} }
if(des.bits.dig == DFHack::designation_no)
des.bits.dig = DFHack::designation_default;
L->setDesignationAt(current,des);
} }
} }
L->WriteAll(); L->WriteAll();