minecraftvulkan/src/chunk.cpp

184 lines
4.9 KiB
C++
Raw Normal View History

2022-09-25 23:05:56 +00:00
#include "chunk.hpp"
namespace app {
static std::map<std::pair<int32_t, int32_t>, Chunk*> chunks{};
2022-09-25 23:05:56 +00:00
Chunk::Chunk(int32_t gridX, int32_t gridZ, uint32_t world_seed)
2022-09-25 23:05:56 +00:00
: world_seed{world_seed},
chunk_seed{(world_seed * gridX) + (world_seed * gridZ) / 2},
gridX{gridX},
gridZ{gridZ} {
generate();
}
Chunk* Chunk::newChunk(int32_t gridX, int32_t gridZ, uint32_t world_seed) {
2022-09-26 15:03:27 +00:00
Chunk* chunk = new Chunk(gridX, gridZ, world_seed);
chunks[{gridX, gridZ}] = std::move(chunk);
return chunks[{gridX, gridZ}];
2022-09-25 23:05:56 +00:00
}
Chunk* Chunk::getChunk(int32_t gridX, int32_t gridZ) {
2022-09-26 15:03:27 +00:00
if(chunks.count({gridX, gridZ})) {
return chunks[{gridX, gridZ}];
} else {
return NULL;
2022-09-25 23:05:56 +00:00
}
}
uint8_t Chunk::getBlock(int32_t x, int32_t y, int32_t z) {
2022-09-26 15:03:27 +00:00
if(y > 256) return AIR;
if(y < 0) return INVALID;
2022-09-25 23:05:56 +00:00
int chunkX = gridX;
int chunkZ = gridZ;
if(x < 0) {
chunkX--;
2022-09-26 15:03:27 +00:00
} else if(x > 15) {
chunkX++;
2022-09-25 23:05:56 +00:00
}
if(z < 0) {
chunkZ--;
2022-09-26 15:03:27 +00:00
} else if(z > 15) {
chunkZ++;
2022-09-25 23:05:56 +00:00
}
x = (x+16)%16;
z = (z+16)%16;
2022-09-25 23:05:56 +00:00
if(chunkX == gridX && chunkZ == gridZ) {
2022-09-26 15:03:27 +00:00
int index = x + (z * 16) + (y * 256);
return cubes[index];
2022-09-25 23:05:56 +00:00
} else {
Chunk* chunk = getChunk(chunkX, chunkZ);
if(chunk == NULL) {
2022-09-26 15:03:27 +00:00
return INVALID;
2022-09-25 23:05:56 +00:00
} else {
2022-09-26 15:03:27 +00:00
int index = x + (z * 16) + (y * 256);
return chunk->cubes[index];
2022-09-25 23:05:56 +00:00
}
}
}
static std::map<uint8_t, Block> blocks{};
static std::map<std::string, uint32_t> texturesIds{};
static std::vector<xe::Image*> textures{};
void loadTexture(const std::string& filePath) {
xe::Image* image = xe::Engine::getInstance()->loadImageFromFile(filePath);
texturesIds[filePath] = static_cast<uint32_t>(textures.size());
textures.push_back(image);
}
uint32_t getTexture(const std::string& filePath) {
return texturesIds[filePath];
}
std::vector<xe::Image*>& Chunk::getTextures() {
return textures;
2022-09-25 23:05:56 +00:00
}
void Chunk::load() {
loadTexture(DIRT_TEXTURE);
loadTexture(GRASS_TEXTURE);
loadTexture(GRASS_TOP_TEXTURE);
blocks[DIRT] = {{getTexture(DIRT_TEXTURE), getTexture(DIRT_TEXTURE), getTexture(DIRT_TEXTURE), getTexture(DIRT_TEXTURE), getTexture(DIRT_TEXTURE), getTexture(DIRT_TEXTURE)}};
blocks[GRASS] = {{getTexture(GRASS_TEXTURE), getTexture(GRASS_TEXTURE), getTexture(GRASS_TOP_TEXTURE), getTexture(DIRT_TEXTURE), getTexture(GRASS_TEXTURE), getTexture(GRASS_TEXTURE)}};
}
void Chunk::unload() {
for(const auto &image: textures) {
delete image;
}
2022-09-26 15:03:27 +00:00
for(const auto &[key, chunk]: chunks) {
delete chunk;
}
chunks.clear();
textures.clear();
}
void Chunk::setBlock(int32_t x, int32_t y, int32_t z, uint8_t block) {
int index = x + (z * 16) + (y * 256);
cubes[index] = block;
2022-09-26 15:03:27 +00:00
}
std::shared_ptr<xe::Model> Chunk::getMesh() {
if(reloadRequired) {
delete chunkMesh.get();
xe::Model::Builder builder{};
builder.vertexData = vertexData;
2022-09-26 22:03:07 +00:00
builder.vertexSize = 36;
2022-09-26 15:03:27 +00:00
chunkMesh = std::make_shared<xe::Model>(xe::Engine::getInstance()->getDevice(), builder);
}
return chunkMesh;
2022-09-25 23:05:56 +00:00
}
void Chunk::createMeshAsync() {
if(working) return;
// worker = std::thread(createMesh);
}
void Chunk::createMesh() {
working = true;
vertexData.data.clear();
for(int32_t x=0;x<16;x++) {
for(int32_t y=0; y<256; y++) {
for(int32_t z=0; z<16; z++) {
2022-09-25 23:05:56 +00:00
uint8_t block = getBlock(x,y,z);
if(block == AIR) continue;
if(getBlock(x+1,y,z) == AIR) {
addVerticies(0, x, y, z, block);
2022-09-25 23:05:56 +00:00
}
if(getBlock(x-1,y,z) == AIR) {
addVerticies(1, x, y, z, block);
2022-09-25 23:05:56 +00:00
}
if(getBlock(x,y+1,z) == AIR) {
addVerticies(2, x, y, z, block);
2022-09-25 23:05:56 +00:00
}
if(getBlock(x,y-1,z) == AIR) {
addVerticies(3, x, y, z, block);
2022-09-25 23:05:56 +00:00
}
if(getBlock(x,y,z+1) == AIR) {
addVerticies(4, x, y, z, block);
2022-09-25 23:05:56 +00:00
}
if(getBlock(x,y,z-1) == AIR) {
addVerticies(5, x, y, z, block);
2022-09-25 23:05:56 +00:00
}
}
}
}
working = false;
reloadRequired = true;
}
void Chunk::addVerticies(uint8_t side, int32_t x, int32_t y, int32_t z, uint8_t block) {
2022-09-25 23:05:56 +00:00
for(int i = 0; i < 6; i ++) {
vertexData.write<float>(px[side * 6 + i][0] + x);
vertexData.write<float>(px[side * 6 + i][1] + y);
vertexData.write<float>(px[side * 6 + i][2] + z);
vertexData.write<float>(nm[side][0]);
vertexData.write<float>(nm[side][1]);
vertexData.write<float>(nm[side][2]);
vertexData.write<float>(uv[i][0]);
vertexData.write<float>(uv[i][1]);
vertexData.write<uint32_t>(static_cast<uint32_t>(blocks[block].textures[side]));
2022-09-25 23:05:56 +00:00
}
}
void Chunk::generate() {
cubes.resize(16*16*256);
2022-09-25 23:05:56 +00:00
2022-09-26 15:03:27 +00:00
const PerlinNoise perlin{world_seed};
2022-09-25 23:05:56 +00:00
for(int x = 0; x < 16; x++) {
for(int z = 0; z < 16; z++) {
2022-09-26 15:03:27 +00:00
int height = perlin.octave2D_01((( x + gridX * 16) * 0.01), ((z + gridZ * 16) * 0.01), 4) * 10;
2022-09-25 23:05:56 +00:00
for(int y = 0; y < 256; y++) {
if(y == height){
setBlock(x, y, z, GRASS);
} else if(y < height)
2022-09-25 23:05:56 +00:00
setBlock(x, y, z, DIRT);
else
setBlock(x, y, z, AIR);
}
}
}
}
}