172 lines
4.5 KiB
C
Raw Normal View History

2013-08-29 11:45:22 +09:00
/*
Copyright (c) 2013 yvt
2013-09-05 00:52:03 +09:00
based on code of pysnip (c) Mathias Kaerlev 2011-2012.
2013-08-29 11:45:22 +09:00
This file is part of OpenSpades.
2013-08-29 11:45:22 +09:00
OpenSpades is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
2013-08-29 11:45:22 +09:00
OpenSpades is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
2013-08-29 11:45:22 +09:00
You should have received a copy of the GNU General Public License
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
2013-08-29 11:45:22 +09:00
*/
2013-08-18 16:18:06 +09:00
#pragma once
#include <cstdint>
#include <functional>
#include <list>
#include <Core/Debug.h>
#include <Core/Math.h>
2013-08-18 16:18:06 +09:00
#include "IGameMapListener.h"
#include <Core/AutoLocker.h>
#include <Core/Mutex.h>
#include <Core/RefCountedObject.h>
2013-08-18 16:18:06 +09:00
namespace spades {
2013-08-18 16:18:06 +09:00
class IStream;
namespace client {
class GameMap : public RefCountedObject {
protected:
~GameMap();
2013-08-18 16:18:06 +09:00
public:
// fixed for now
enum {
DefaultWidth = 512,
DefaultHeight = 512,
DefaultDepth = 64 // should be <= 64
};
GameMap();
/**
* Construct a `GameMap` from VOXLAP5 terrain data supplied by the specified stream.
*
* @param onProgress Called whenever a new column (a set of voxels with the same X and Y
* coordinates) is loaded from the stream. The parameter indicates
* the number of columns loaded
* (up to `DefaultWidth * DefaultHeight`).
*/
static GameMap *Load(IStream *, std::function<void(int)> onProgress = {});
2013-11-22 19:34:20 +09:00
void Save(IStream *);
2013-08-18 16:18:06 +09:00
int Width() { return DefaultWidth; }
int Height() { return DefaultHeight; }
int Depth() { return DefaultDepth; }
inline bool IsSolid(int x, int y, int z) {
SPAssert(x >= 0);
SPAssert(x < Width());
SPAssert(y >= 0);
SPAssert(y < Height());
SPAssert(z >= 0);
SPAssert(z < Depth());
2013-08-18 16:18:06 +09:00
return ((solidMap[x][y] >> (uint64_t)z) & 1ULL) != 0;
}
2013-08-18 16:18:06 +09:00
/** @return 0xHHBBGGRR where HH is health (up to 100) */
inline uint32_t GetColor(int x, int y, int z) {
SPAssert(x >= 0);
SPAssert(x < Width());
SPAssert(y >= 0);
SPAssert(y < Height());
SPAssert(z >= 0);
SPAssert(z < Depth());
2013-08-18 16:18:06 +09:00
return colorMap[x][y][z];
}
inline uint64_t GetSolidMapWrapped(int x, int y) {
return solidMap[x & (Width() - 1)][y & (Height() - 1)];
}
inline bool IsSolidWrapped(int x, int y, int z) {
if (z < 0)
2013-08-18 16:18:06 +09:00
return false;
if (z >= Depth())
2013-08-18 16:18:06 +09:00
return true;
return ((solidMap[x & (Width() - 1)][y & (Height() - 1)] >> (uint64_t)z) & 1ULL) !=
0;
2013-08-18 16:18:06 +09:00
}
inline uint32_t GetColorWrapped(int x, int y, int z) {
return colorMap[x & (Width() - 1)][y & (Height() - 1)][z & (Depth() - 1)];
2013-08-18 16:18:06 +09:00
}
inline void Set(int x, int y, int z, bool solid, uint32_t color, bool unsafe = false) {
SPAssert(x >= 0);
SPAssert(x < Width());
SPAssert(y >= 0);
SPAssert(y < Height());
SPAssert(z >= 0);
SPAssert(z < Depth());
2013-08-18 16:18:06 +09:00
uint64_t mask = 1ULL << z;
uint64_t value = solidMap[x][y];
bool changed = false;
if ((value & mask) != (solid ? mask : 0ULL)) {
2013-08-18 16:18:06 +09:00
changed = true;
value &= ~mask;
if (solid)
2013-08-18 16:18:06 +09:00
value |= mask;
solidMap[x][y] = value;
}
if (solid) {
if (color != colorMap[x][y][z]) {
2013-08-18 16:18:06 +09:00
changed = true;
colorMap[x][y][z] = color;
}
}
if (!unsafe) {
if (changed) {
{
AutoLocker guard(&listenersMutex);
for (auto *l : listeners) {
l->GameMapChanged(x, y, z, this);
}
}
}
2013-08-18 16:18:06 +09:00
}
}
void AddListener(IGameMapListener *);
void RemoveListener(IGameMapListener *);
2013-08-18 16:18:06 +09:00
bool ClipBox(int x, int y, int z);
bool ClipWorld(int x, int y, int z);
2013-08-18 16:18:06 +09:00
bool ClipBox(float x, float y, float z);
bool ClipWorld(float x, float y, float z);
2013-08-18 16:18:06 +09:00
// vanila compat
bool CastRay(Vector3 v0, Vector3 v1, float length, IntVector3 &vOut);
2013-08-18 16:18:06 +09:00
// accurate and slow ray casting
struct RayCastResult {
bool hit;
bool startSolid;
Vector3 hitPos;
IntVector3 hitBlock;
IntVector3 normal;
};
RayCastResult CastRay2(Vector3 v0, Vector3 dir, int maxSteps);
2013-08-18 16:18:06 +09:00
private:
uint64_t solidMap[DefaultWidth][DefaultHeight];
uint32_t colorMap[DefaultWidth][DefaultHeight][DefaultDepth];
std::list<IGameMapListener *> listeners;
Mutex listenersMutex;
2013-11-22 19:34:20 +09:00
bool IsSurface(int x, int y, int z);
2013-08-18 16:18:06 +09:00
};
}
}