2009-04-15 14:03:31 +00:00
|
|
|
/*
|
2014-05-26 10:37:04 +02:00
|
|
|
* int3.h, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
2017-07-13 11:26:03 +03:00
|
|
|
#pragma once
|
2009-04-16 11:14:13 +00:00
|
|
|
|
2022-07-26 16:07:42 +03:00
|
|
|
VCMI_LIB_NAMESPACE_BEGIN
|
|
|
|
|
2011-02-22 11:52:36 +00:00
|
|
|
/// Class which consists of three integer values. Represents position on adventure map.
|
2009-04-16 11:14:13 +00:00
|
|
|
class int3
|
|
|
|
{
|
|
|
|
public:
|
2014-05-25 20:12:53 +02:00
|
|
|
si32 x, y, z;
|
|
|
|
|
2014-05-26 10:37:04 +02:00
|
|
|
//c-tor: x, y, z initialized to 0
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3() : x(0), y(0), z(0) {} // I think that x, y, z should be left uninitialized.
|
2014-05-26 10:37:04 +02:00
|
|
|
//c-tor: x, y, z initialized to i
|
2023-02-12 04:03:04 +03:00
|
|
|
explicit constexpr int3(const si32 i) : x(i), y(i), z(i) {}
|
2014-05-26 10:37:04 +02:00
|
|
|
//c-tor: x, y, z initialized to X, Y, Z
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3(const si32 X, const si32 Y, const si32 Z) : x(X), y(Y), z(Z) {}
|
|
|
|
constexpr int3(const int3 & c) = default;
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 & operator=(const int3 & c) = default;
|
|
|
|
constexpr int3 operator-() const { return int3(-x, -y, -z); }
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 operator+(const int3 & i) const { return int3(x + i.x, y + i.y, z + i.z); }
|
|
|
|
constexpr int3 operator-(const int3 & i) const { return int3(x - i.x, y - i.y, z - i.z); }
|
2014-05-26 10:37:04 +02:00
|
|
|
//returns int3 with coordinates increased by given number
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 operator+(const si32 i) const { return int3(x + i, y + i, z + i); }
|
2014-05-26 10:37:04 +02:00
|
|
|
//returns int3 with coordinates decreased by given number
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 operator-(const si32 i) const { return int3(x - i, y - i, z - i); }
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2021-05-16 15:39:38 +03:00
|
|
|
//returns int3 with coordinates multiplied by given number
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 operator*(const double i) const { return int3((int)(x * i), (int)(y * i), (int)(z * i)); }
|
2021-05-16 15:39:38 +03:00
|
|
|
//returns int3 with coordinates divided by given number
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 operator/(const double i) const { return int3((int)(x / i), (int)(y / i), (int)(z / i)); }
|
2021-05-16 15:39:38 +03:00
|
|
|
|
|
|
|
//returns int3 with coordinates multiplied by given number
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 operator*(const si32 i) const { return int3(x * i, y * i, z * i); }
|
2021-05-16 15:39:38 +03:00
|
|
|
//returns int3 with coordinates divided by given number
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 operator/(const si32 i) const { return int3(x / i, y / i, z / i); }
|
2021-05-16 15:39:38 +03:00
|
|
|
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 & operator+=(const int3 & i)
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
|
|
|
x += i.x;
|
|
|
|
y += i.y;
|
|
|
|
z += i.z;
|
|
|
|
return *this;
|
|
|
|
}
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 & operator-=(const int3 & i)
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
|
|
|
x -= i.x;
|
|
|
|
y -= i.y;
|
|
|
|
z -= i.z;
|
|
|
|
return *this;
|
|
|
|
}
|
2017-08-11 20:03:05 +03:00
|
|
|
|
2014-05-26 10:37:04 +02:00
|
|
|
//increases all coordinates by given number
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 & operator+=(const si32 i)
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
|
|
|
x += i;
|
|
|
|
y += i;
|
|
|
|
z += i;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
//decreases all coordinates by given number
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr int3 & operator-=(const si32 i)
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
|
|
|
x -= i;
|
|
|
|
y -= i;
|
|
|
|
z -= i;
|
|
|
|
return *this;
|
|
|
|
}
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr bool operator==(const int3 & i) const { return (x == i.x && y == i.y && z == i.z); }
|
|
|
|
constexpr bool operator!=(const int3 & i) const { return (x != i.x || y != i.y || z != i.z); }
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr bool operator<(const int3 & i) const
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
|
|
|
if (z < i.z)
|
|
|
|
return true;
|
|
|
|
if (z > i.z)
|
|
|
|
return false;
|
|
|
|
if (y < i.y)
|
|
|
|
return true;
|
|
|
|
if (y > i.y)
|
|
|
|
return false;
|
|
|
|
if (x < i.x)
|
|
|
|
return true;
|
|
|
|
if (x > i.x)
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2016-02-10 18:36:56 +03:00
|
|
|
enum EDistanceFormula
|
|
|
|
{
|
|
|
|
DIST_2D = 0,
|
|
|
|
DIST_MANHATTAN, // patrol distance
|
|
|
|
DIST_CHEBYSHEV, // ambient sound distance
|
|
|
|
DIST_2DSQ
|
|
|
|
};
|
|
|
|
|
|
|
|
ui32 dist(const int3 & o, EDistanceFormula formula) const
|
|
|
|
{
|
|
|
|
switch(formula)
|
|
|
|
{
|
|
|
|
case DIST_2D:
|
2019-01-19 13:52:02 +03:00
|
|
|
return static_cast<ui32>(dist2d(o));
|
2016-02-10 18:36:56 +03:00
|
|
|
case DIST_MANHATTAN:
|
2019-01-19 13:52:02 +03:00
|
|
|
return static_cast<ui32>(mandist2d(o));
|
2016-02-10 18:36:56 +03:00
|
|
|
case DIST_CHEBYSHEV:
|
2019-01-19 13:52:02 +03:00
|
|
|
return static_cast<ui32>(chebdist2d(o));
|
2016-02-10 18:36:56 +03:00
|
|
|
case DIST_2DSQ:
|
|
|
|
return dist2dSQ(o);
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-26 10:37:04 +02:00
|
|
|
//returns squared distance on Oxy plane (z coord is not used)
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr ui32 dist2dSQ(const int3 & o) const
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
|
|
|
const si32 dx = (x - o.x);
|
|
|
|
const si32 dy = (y - o.y);
|
2014-08-30 18:45:11 +02:00
|
|
|
return (ui32)(dx*dx) + (ui32)(dy*dy);
|
2014-05-26 10:37:04 +02:00
|
|
|
}
|
|
|
|
//returns distance on Oxy plane (z coord is not used)
|
|
|
|
double dist2d(const int3 & o) const
|
|
|
|
{
|
|
|
|
return std::sqrt((double)dist2dSQ(o));
|
|
|
|
}
|
2015-12-04 21:08:09 +03:00
|
|
|
//manhattan distance used for patrol radius (z coord is not used)
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr double mandist2d(const int3 & o) const
|
2015-12-04 21:08:09 +03:00
|
|
|
{
|
2023-02-12 04:03:04 +03:00
|
|
|
return vstd::abs(o.x - x) + vstd::abs(o.y - y);
|
2015-12-04 21:08:09 +03:00
|
|
|
}
|
2016-02-10 18:36:56 +03:00
|
|
|
//chebyshev distance used for ambient sounds (z coord is not used)
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr double chebdist2d(const int3 & o) const
|
2016-02-10 18:36:56 +03:00
|
|
|
{
|
2023-02-12 04:03:04 +03:00
|
|
|
return std::max(vstd::abs(o.x - x), vstd::abs(o.y - y));
|
2016-02-10 18:36:56 +03:00
|
|
|
}
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr bool areNeighbours(const int3 & o) const
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
|
|
|
return (dist2dSQ(o) < 4) && (z == o.z);
|
|
|
|
}
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2014-05-26 10:37:04 +02:00
|
|
|
//returns "(x y z)" string
|
2017-08-11 20:03:05 +03:00
|
|
|
std::string toString() const
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
2022-09-14 11:00:40 +02:00
|
|
|
//Performance is important here
|
2022-12-26 14:48:13 +02:00
|
|
|
std::string result = "(" +
|
2022-12-25 14:04:12 +02:00
|
|
|
std::to_string(x) + " " +
|
|
|
|
std::to_string(y) + " " +
|
2022-12-26 14:48:13 +02:00
|
|
|
std::to_string(z) + ")";
|
2022-09-14 11:00:40 +02:00
|
|
|
|
2022-12-25 14:04:12 +02:00
|
|
|
return result;
|
2014-05-26 10:37:04 +02:00
|
|
|
}
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr bool valid() const //Should be named "isValid"?
|
2014-05-26 10:37:04 +02:00
|
|
|
{
|
|
|
|
return z >= 0; //minimal condition that needs to be fulfilled for tiles in the map
|
|
|
|
}
|
2014-05-25 20:12:53 +02:00
|
|
|
|
|
|
|
template <typename Handler>
|
2014-05-26 10:37:04 +02:00
|
|
|
void serialize(Handler &h, const int version)
|
|
|
|
{
|
2017-07-31 16:35:42 +03:00
|
|
|
h & x;
|
|
|
|
h & y;
|
|
|
|
h & z;
|
2014-05-26 10:37:04 +02:00
|
|
|
}
|
2015-12-04 01:06:02 +02:00
|
|
|
|
2023-02-12 04:03:04 +03:00
|
|
|
constexpr static std::array<int3, 8> getDirs()
|
2015-12-04 01:06:02 +02:00
|
|
|
{
|
|
|
|
return { { int3(0,1,0),int3(0,-1,0),int3(-1,0,0),int3(+1,0,0),
|
|
|
|
int3(1,1,0),int3(-1,1,0),int3(1,-1,0),int3(-1,-1,0) } };
|
|
|
|
}
|
2009-04-16 11:14:13 +00:00
|
|
|
};
|
2014-05-25 20:12:53 +02:00
|
|
|
|
2014-06-18 09:57:36 +02:00
|
|
|
template<typename Container>
|
|
|
|
int3 findClosestTile (Container & container, int3 dest)
|
|
|
|
{
|
2014-08-30 19:08:04 +02:00
|
|
|
static_assert(std::is_same<typename Container::value_type, int3>::value,
|
2014-08-30 18:45:11 +02:00
|
|
|
"findClosestTile requires <int3> container.");
|
|
|
|
|
|
|
|
int3 result(-1, -1, -1);
|
2014-06-18 09:57:36 +02:00
|
|
|
ui32 distance = std::numeric_limits<ui32>::max();
|
2014-08-30 18:45:11 +02:00
|
|
|
for (const int3& tile : container)
|
2014-06-18 09:57:36 +02:00
|
|
|
{
|
2014-08-30 18:45:11 +02:00
|
|
|
const ui32 currentDistance = dest.dist2dSQ(tile);
|
2014-06-18 09:57:36 +02:00
|
|
|
if (currentDistance < distance)
|
|
|
|
{
|
|
|
|
result = tile;
|
|
|
|
distance = currentDistance;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
2015-12-04 01:06:02 +02:00
|
|
|
}
|
2022-07-26 16:07:42 +03:00
|
|
|
|
|
|
|
VCMI_LIB_NAMESPACE_END
|
2023-04-16 02:15:12 +03:00
|
|
|
|
|
|
|
|
|
|
|
template<>
|
|
|
|
struct std::hash<VCMI_LIB_WRAP_NAMESPACE(int3)> {
|
|
|
|
size_t operator()(VCMI_LIB_WRAP_NAMESPACE(int3) const& pos) const
|
|
|
|
{
|
|
|
|
size_t ret = std::hash<int>()(pos.x);
|
|
|
|
VCMI_LIB_WRAP_NAMESPACE(vstd)::hash_combine(ret, pos.y);
|
|
|
|
VCMI_LIB_WRAP_NAMESPACE(vstd)::hash_combine(ret, pos.z);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
};
|