diff --git a/.gitignore b/.gitignore index de160309..2556a808 100644 --- a/.gitignore +++ b/.gitignore @@ -67,4 +67,6 @@ _deps # Custom build .vscode -lib \ No newline at end of file +lib + +assets/tileset.png \ No newline at end of file diff --git a/assets/shaders/test.frag b/assets/shaders/test.frag index dc3182c1..69aac984 100644 --- a/assets/shaders/test.frag +++ b/assets/shaders/test.frag @@ -8,6 +8,9 @@ out vec4 FragColor; void main() { vec4 color = texture(u_Text, TexCoord); - // FragColor = color; - FragColor = vec4(1, 1, 1, 1); + if(color.a == 0) { + FragColor = vec4(1, 1, 1, 1); + } else { + FragColor = color; + } } \ No newline at end of file diff --git a/src/engine/display/tileset.c b/src/engine/display/tileset.c index 904f4b57..92e03ef5 100644 --- a/src/engine/display/tileset.c +++ b/src/engine/display/tileset.c @@ -9,17 +9,15 @@ tileset_t * tilesetCreate( int32_t columns, int32_t rows, + int32_t width, int32_t height, int32_t gapX, int32_t gapY, - int32_t borderX, int32_t borderY, - int32_t width, int32_t height + int32_t borderX, int32_t borderY ) { tileset_t *tileset; float divX, divY, tdivX; - int32_t x, y, i, j, count; - + int32_t x, y, i, count; tileset = malloc(sizeof(tileset_t)); - if(tileset == NULL) return NULL; count = rows * columns; diff --git a/src/engine/display/tileset.h b/src/engine/display/tileset.h index bcf9d85f..bad980ec 100644 --- a/src/engine/display/tileset.h +++ b/src/engine/display/tileset.h @@ -27,19 +27,19 @@ typedef struct { * * @param columns Count of columns. * @param rows Count of rows. + * @param width Width of the tileset. + * @param height Height of the tileset. * @param gapX Space between each column. * @param gapY Space between each row. * @param borderX Space around the edge of the tileset. * @param borderY Space around the edge of the tileset. - * @param width Width of the tileset. - * @param height Height of the tileset. * @returns The tileset. */ tileset_t * tilesetCreate( int32_t columns, int32_t rows, + int32_t width, int32_t height, int32_t gapX, int32_t gapY, - int32_t borderX, int32_t borderY, - int32_t width, int32_t height + int32_t borderX, int32_t borderY ); /** diff --git a/src/engine/engine.c b/src/engine/engine.c index 7df15eea..698794e0 100644 --- a/src/engine/engine.c +++ b/src/engine/engine.c @@ -1,16 +1,18 @@ /** - * Copyright (c) 2021 Dominic Msters + * Copyright (c) 2021 Dominic Masters * * This software is released under the MIT License. * https://opensource.org/licenses/MIT */ - #include "engine.h" camera_t *camera; shader_t *shader; world_t *world; + +texture_t *texture; spritebatch_t *batch; +tileset_t *tileset; int i; @@ -47,6 +49,13 @@ engine_t * engineInit(platform_t *platform, char *name, uint32_t inputCount) { shaderUseCamera(shader, camera); world = worldCreate(); + texture = assetTextureLoad("tileset.png"); + tileset = tilesetCreate( + texture->width/16, texture->height/16, + texture->width, texture->height, + 0, 0, + 0, 0 + ); i = 0; return engine; diff --git a/src/engine/engine.h b/src/engine/engine.h index 2ee4f9fc..13a0bf7f 100644 --- a/src/engine/engine.h +++ b/src/engine/engine.h @@ -15,6 +15,8 @@ #include "world/world.h" #include "world/chunklist.h" #include "display/spritebatch.h" +#include "display/tileset.h" +#include "display/texture.h" /** Information about the current engine context. */ diff --git a/src/engine/texture.h b/src/engine/texture.h new file mode 100644 index 00000000..79ede17b --- /dev/null +++ b/src/engine/texture.h @@ -0,0 +1,57 @@ +// Copyright (c) 2021 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include +#include +#include +#include + +/** + * Structure detailing information about a texture. + * Because we plan to upload the pixels of a texture into the GPU, we don't + * store the pixels in memory because we don't need to! + */ +typedef struct { + int32_t width; + int32_t height; + GLuint id; +} texture_t; + +/** Information about a single pixel. */ +typedef struct { + uint8_t r, g, b, a ; +} pixel_t; + +/** + * Creates a new texture that can be written in to. + * + * @param width Width of the texture (in pixels). + * @param height Height of the texture (in pixels). + * @param pixels Default pixel array, set to NULL to set all pixel data to 0. + * @return The newly created texture instance. + */ +texture_t * textureCreate(int32_t width, int32_t height, pixel_t *pixels); + +/** + * Buffer pixel data onto the GPU. Pixel buffering is rather costly so avoid + * doing this too often. + * + * @param texture Texture to buffer in to. + * @param x X coordinate in texture space to render to. + * @param y Y coordinate in texture space to render to. + * @param width Width of the pixel region you're buffering. + * @param height Height of the pixel region you're buffering. + * @param pixels Array of pixels to buffer onto the GPU. + */ +void textureBufferPixels(texture_t *texture, + int32_t x, int32_t y, int32_t width, int32_t height, pixel_t *pixels +); + +/** + * Clean a previously created texture. + * @param texture Texture to clean up. + */ +void textureDispose(texture_t *texture); \ No newline at end of file diff --git a/src/engine/world/chunk.c b/src/engine/world/chunk.c index 8d14350a..d32496ed 100644 --- a/src/engine/world/chunk.c +++ b/src/engine/world/chunk.c @@ -4,7 +4,6 @@ * This software is released under the MIT License. * https://opensource.org/licenses/MIT */ - #include "chunk.h" void chunkCreate(chunk_t *chunk) { @@ -12,7 +11,6 @@ void chunkCreate(chunk_t *chunk) { count = CHUNK_WIDTH * CHUNK_HEIGHT * CHUNK_DEPTH; chunk->batch = spriteBatchCreate(count); - chunk->tiles = malloc(sizeof(tile_t) * count); } void chunkRender(chunk_t *chunk, shader_t *shader) { @@ -25,24 +23,4 @@ void chunkRender(chunk_t *chunk, shader_t *shader) { // Render the batch. shaderUsePosition(shader, x, y, z, 0, 0, 0); spriteBatchDraw(chunk->batch, 0, -1); -} - -void chunkLoad(chunk_t *chunk, int32_t x, int32_t y, int32_t z) { - int32_t tx, ty, tz; - - spriteBatchFlush(chunk->batch); - - // for(tx = 0; tx < CHUNK_WIDTH; tx++) { - // for(ty = 0; ty < CHUNK_HEIGHT; ty++) { - // for(tz = 0; tz < CHUNK_DEPTH; tz++) { - // spriteBatchQuad(chunk->batch, -1, - // x, y, z, 1, 1, 0, 0, 1, 1 - // ); - // } - // } - // } -} - -void chunkUnload(chunk_t *chunk) { - } \ No newline at end of file diff --git a/src/engine/world/chunk.h b/src/engine/world/chunk.h index 82205a18..c7e49374 100644 --- a/src/engine/world/chunk.h +++ b/src/engine/world/chunk.h @@ -5,23 +5,19 @@ #pragma once #include -#include "../display/primitive.h" +#include #include "../display/shader.h" #include "../display/spritebatch.h" +#include "tile.h" #define CHUNK_WIDTH 3 #define CHUNK_HEIGHT CHUNK_WIDTH #define CHUNK_DEPTH CHUNK_HEIGHT -typedef uint8_t tile_t; - typedef struct { /** Absolute X Y Z chunklist coordinates */ int32_t x, y, z; - /** Tiles within the chunk */ - tile_t *tiles; - /** Sprite Batch for holding static tiles */ spritebatch_t *batch; } chunk_t; @@ -35,22 +31,9 @@ typedef struct { void chunkCreate(chunk_t *chunk); /** - * Loads a given chunk into the memory specified. + * Render a given chunk. * - * @param chunk Chunk to load into. - * @param x X of the chunk. - * @param y Y of the chunk. - * @param z Z of the chunk. + * @param chunk Chunk to render + * @param */ -void chunkLoad(chunk_t *chunk, int32_t x, int32_t y, int32_t z); - -/** - * Unload a given chunk. - * - * @param chunk Chunk to unload. - */ -void chunkUnload(chunk_t *chunk); - - - void chunkRender(chunk_t *chunk, shader_t *shader); \ No newline at end of file diff --git a/src/engine/world/chunklist.c b/src/engine/world/chunklist.c index a64ca831..068eb3e2 100644 --- a/src/engine/world/chunklist.c +++ b/src/engine/world/chunklist.c @@ -7,7 +7,9 @@ #include "chunklist.h" -chunklist_t * chunkListCreate(int32_t width, int32_t height, int32_t depth) { +chunklist_t * chunkListCreate(int32_t width, int32_t height, int32_t depth, + tilemap_t *tilemap +) { chunklist_t *list; chunk_t *chunk; int32_t i, x, y, z; @@ -18,6 +20,7 @@ chunklist_t * chunkListCreate(int32_t width, int32_t height, int32_t depth) { list->width = width, list->height = height, list->depth = depth; list->x = 0, list->y = 0, list->z = 0; list->count = width * height * depth; + list->tilemap = tilemap; //Create chunks list->chunks = malloc(list->count * sizeof(chunk_t)); @@ -47,7 +50,7 @@ chunklist_t * chunkListCreate(int32_t width, int32_t height, int32_t depth) { chunk->z = z; chunkCreate(chunk); - chunkLoad(chunk, x, y, z); + chunkListChunkLoad(list, chunk, x, y, z); i++; } } @@ -105,8 +108,8 @@ void chunkListShift(chunklist_t *list, int32_t x, int32_t y, int32_t z) { az = lz + nz; // Load new chunk. - chunkUnload(chunk); - chunkLoad(chunk, ax, ay, az); + chunkListChunkUnload(list, chunk); + chunkListChunkLoad(list, chunk, ax, ay, az); // Update the absolute coordinates. chunk->x = ax; @@ -117,7 +120,7 @@ void chunkListShift(chunklist_t *list, int32_t x, int32_t y, int32_t z) { // Now, based off those new local positions, calculate the new index. ni = ( nx + - (ny * list->width) + + (ny * list->width) + (nz * wh) ); chunkList[ni] = chunk; @@ -141,4 +144,41 @@ void chunkListAlign(chunklist_t *list, int32_t x, int32_t y, int32_t z) { lz = z - list->z; chunkListShift(list, lx, ly, lz); +} + +void chunkListChunkLoad(chunklist_t *list, chunk_t *chunk, + int32_t x, int32_t y, int32_t z +) { + int32_t tx, ty, tz, i, count; + tilesetdiv_t *div; + tileref_t *tiles; + tileref_t tile; + + count = CHUNK_WIDTH * CHUNK_DEPTH * CHUNK_HEIGHT; + tiles = calloc(count, sizeof(tileref_t)); + tiles[0] = 0x01; + + // Render the quads. + i = -1; + for(tx = 0; tx < CHUNK_WIDTH; tx++) { + for(ty = 0; ty < CHUNK_HEIGHT; ty++) { + for(tz = 0; tz < CHUNK_DEPTH; tz++) { + tile = tiles[++i]; + if(tile == TILE_NULL) continue; + div = &list->tilemap->tileset->divisions[tile]; + + spriteBatchQuad(chunk->batch, -1, + tx, ty, tz, 1, 1, + div->x0, div->y0, div->x1, div->y1 + ); + } + } + } + + free(tiles); +} + +void chunkListChunkUnload(chunklist_t *list, chunk_t *chunk) { + // Flush the sprite batch. + spriteBatchFlush(chunk->batch); } \ No newline at end of file diff --git a/src/engine/world/chunklist.h b/src/engine/world/chunklist.h index 0ec240f7..c78f2b2e 100644 --- a/src/engine/world/chunklist.h +++ b/src/engine/world/chunklist.h @@ -6,8 +6,10 @@ #pragma once #include #include -#include "chunk.h" #include "./../util/math.h" +#include "chunk.h" +#include "tile.h" +#include "../display/texture.h" typedef struct { /** Dimensions of the chunk list */ @@ -22,6 +24,9 @@ typedef struct { /** Current order of each chunk in the list. */ chunk_t **chunkList; + /** Chunk list tile map. */ + tilemap_t *tilemap; + /** The actual chunks in the list */ chunk_t *chunks; } chunklist_t; @@ -33,9 +38,12 @@ typedef struct { * @param width The width (x axis) of chunks to keep loaded. * @param height The height (y axis) of chunks to keep loaded. * @param depth The depth (z axis) of chunks to keep loaded. + * @param tilemap Tilemap of the chunk list. * @return A new chunk list. */ -chunklist_t * chunkListCreate(int32_t width, int32_t height, int32_t depth); +chunklist_t * chunkListCreate(int32_t width, int32_t height, int32_t depth, + tilemap_t *tilemap +); /** * Disposes and frees a previously created list. This does not free the chunks @@ -63,4 +71,25 @@ void chunkListShift(chunklist_t *list, int32_t x, int32_t y, int32_t z); * @param y Y movement to shift chunk along. * @param z Z movement to shift chunk along. */ -void chunkListAlign(chunklist_t *list, int32_t x, int32_t y, int32_t z); \ No newline at end of file +void chunkListAlign(chunklist_t *list, int32_t x, int32_t y, int32_t z); + +/** + * Loads a given chunk. + * + * @param list List to load the chunk for. + * @param chunk The chunk to load. + * @param x X of the chunk. + * @param y Y of the chunk. + * @param z Z of the chunk. + */ +void chunkListChunkLoad(chunklist_t *list, chunk_t *chunk, + int32_t x, int32_t y, int32_t z +); + +/** + * Unload a given chunk. + * + * @param list List that the chunk belongs to. + * @param chunk Chunk to unload. + */ +void chunkListChunkUnload(chunklist_t *list, chunk_t *chunk); \ No newline at end of file diff --git a/src/engine/world/tile.c b/src/engine/world/tile.c new file mode 100644 index 00000000..82e32895 --- /dev/null +++ b/src/engine/world/tile.c @@ -0,0 +1,22 @@ +/** + * Copyright (c) 2021 Dominic Masters + * + * This software is released under the MIT License. + * https://opensource.org/licenses/MIT + */ + +#include "tile.h" + +tilemap_t * tileMapCreate(tileset_t *tileset) { + tilemap_t *tilemap = malloc(sizeof(tilemap_t)); + if(tilemap == NULL) return NULL; + + tilemap->tileset = tileset; + tilemap->tiles = malloc(sizeof(tile_t) * tileset->columns * tileset->rows); + + return tilemap; +} + +void tileMapDispose(tilemap_t *tilemap) { + free(tilemap); +} \ No newline at end of file diff --git a/src/engine/world/tile.h b/src/engine/world/tile.h new file mode 100644 index 00000000..d662a94c --- /dev/null +++ b/src/engine/world/tile.h @@ -0,0 +1,41 @@ +// Copyright (c) 2021 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include +#include +#include "../display/tileset.h" + +typedef uint8_t tileflag_t; +typedef uint16_t tileref_t; + +typedef struct { + tileflag_t flags; +} tile_t; + +typedef struct { + tile_t *tiles; + tileset_t *tileset; +} tilemap_t; + +#define TILE_NULL (tileref_t)0 +#define TILE_FLAG_DYNAMIC (tileflag_t)1 +#define TILE_FLAG_3D (tileflag_t)2 +#define TILE_FLAG_SOLID (tileflag_t)4 + +/** + * Creates a tilemap from a tileset. + * + * @param tileset Tileset to create from. + * @returns The tilemap. + */ +tilemap_t * tileMapCreate(tileset_t *tileset); + +/** + * Destroys a previously created tilemap. + * + * @param tilemap Tilemap to dispose. + */ +void tileMapDispose(tilemap_t *tilemap); \ No newline at end of file diff --git a/src/engine/world/world.c b/src/engine/world/world.c index 73cf20e5..5a84eee0 100644 --- a/src/engine/world/world.c +++ b/src/engine/world/world.c @@ -11,8 +11,40 @@ world_t * worldCreate() { world_t *world = malloc(sizeof(world_t)); if(world == NULL) return NULL; - world->chunkList = chunkListCreate(3, 3, 3); + // Texture + world->texture = assetTextureLoad("tileset.png"); + if(world->texture == NULL) { + free(world); + return NULL; + } + + // Tileset + world->tileset = tilesetCreate( + world->texture->width / 16, world->texture->height / 16, + world->texture->width, world->texture->height, + 0, 0, 0, 0, 0 + ); + if(world->tileset == NULL) { + textureDispose(world->texture); + free(world); + return NULL; + } + + // Tilemap + world->tilemap = tileMapCreate(world->tileset); + if(world->tilemap == NULL) { + textureDispose(world->texture); + tilesetDispose(world->tileset); + free(world); + return NULL; + } + + // Chunk Lists + world->chunkList = chunkListCreate(3, 3, 3, world->tilemap); if(world->chunkList == NULL) { + textureDispose(world->texture); + tilesetDispose(world->tileset); + tileMapDispose(world->tilemap); free(world); return NULL; } @@ -22,6 +54,9 @@ world_t * worldCreate() { void worldRender(world_t *world, shader_t *shader) { int32_t i; + + shaderUseTexture(shader, world->texture); + for(i = 0; i < world->chunkList->count; i++) { chunkRender(world->chunkList->chunks + i, shader); } diff --git a/src/engine/world/world.h b/src/engine/world/world.h index 227fd30a..a4bdc145 100644 --- a/src/engine/world/world.h +++ b/src/engine/world/world.h @@ -5,10 +5,17 @@ #pragma once #include "chunklist.h" +#include "tile.h" #include "../display/shader.h" +#include "../display/tileset.h" +#include "../display/texture.h" +#include "../file/asset.h" typedef struct { chunklist_t *chunkList; + texture_t *texture; + tileset_t *tileset; + tilemap_t *tilemap; } world_t; world_t * worldCreate();