Refactored some tooling.
This commit is contained in:
@ -9,13 +9,24 @@ set(CMAKE_C_STANDARD 99)
|
|||||||
set(CMAKE_C_STANDARD_REQUIRED ON)
|
set(CMAKE_C_STANDARD_REQUIRED ON)
|
||||||
project(Dawn VERSION 1.0)
|
project(Dawn VERSION 1.0)
|
||||||
|
|
||||||
set(GAME_NAME DawnGame)
|
# Targets
|
||||||
set(GAME_VERSION 1.0)
|
if(TARGET_GAME STREQUAL poker)
|
||||||
|
add_compile_definitions(
|
||||||
|
SETTING_GAME_NAME="Penny's Poker"
|
||||||
|
GAME_FILE="poker/game.h"
|
||||||
|
GAME_TYPE=pokergame_t
|
||||||
|
GAME_INIT=pokerGameInit
|
||||||
|
GAME_UPDATE=pokerGameUpdate
|
||||||
|
GAME_DISPOSE=pokerGameDispose
|
||||||
|
GAME_VERSION=1.0
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
##################################### LIBS #####################################
|
# Shared
|
||||||
add_subdirectory(lib)
|
add_subdirectory(lib)
|
||||||
add_subdirectory(src)
|
add_subdirectory(src)
|
||||||
|
|
||||||
|
# Targets
|
||||||
if(TARGET_GROUP STREQUAL test)
|
if(TARGET_GROUP STREQUAL test)
|
||||||
add_subdirectory(test)
|
add_subdirectory(test)
|
||||||
else()
|
else()
|
||||||
|
@ -52,7 +52,6 @@ int32_t main() {
|
|||||||
renderSetResolution(&game->engine.render,
|
renderSetResolution(&game->engine.render,
|
||||||
WINDOW_WIDTH_DEFAULT, WINDOW_HEIGHT_DEFAULT
|
WINDOW_WIDTH_DEFAULT, WINDOW_HEIGHT_DEFAULT
|
||||||
);
|
);
|
||||||
|
|
||||||
// Init the game
|
// Init the game
|
||||||
if(gameInit(game)) {
|
if(gameInit(game)) {
|
||||||
// Bind initial keys
|
// Bind initial keys
|
||||||
@ -72,8 +71,12 @@ int32_t main() {
|
|||||||
// Bind the fake inputs
|
// Bind the fake inputs
|
||||||
inputBind(input, INPUT_MOUSE_X, GLFW_PLATFORM_INPUT_MOUSE_X);
|
inputBind(input, INPUT_MOUSE_X, GLFW_PLATFORM_INPUT_MOUSE_X);
|
||||||
inputBind(input, INPUT_MOUSE_Y, GLFW_PLATFORM_INPUT_MOUSE_Y);
|
inputBind(input, INPUT_MOUSE_Y, GLFW_PLATFORM_INPUT_MOUSE_Y);
|
||||||
glfwSetInputMode(window, GLFW_CURSOR, GLFW_CURSOR_NORMAL);
|
|
||||||
|
|
||||||
|
// Set up some GLFW stuff
|
||||||
|
glfwSetInputMode(window, GLFW_CURSOR, GLFW_CURSOR_NORMAL);
|
||||||
|
glfwSetWindowTitle(window, game->engine.name);
|
||||||
|
|
||||||
|
// Begin time.
|
||||||
time = 0;
|
time = 0;
|
||||||
|
|
||||||
// Main Render Loop
|
// Main Render Loop
|
||||||
|
12
package.json
Normal file
12
package.json
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{
|
||||||
|
"name": "dawn",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"repository": "https://YourWishes@github.com/YourWishes/Dawn.git",
|
||||||
|
"author": "Dominic Masters <dominic@domsplace.com>",
|
||||||
|
"license": "MIT",
|
||||||
|
"private": true,
|
||||||
|
"dependencies": {
|
||||||
|
"pngjs": "^6.0.0",
|
||||||
|
"xml-js": "^1.6.11"
|
||||||
|
}
|
||||||
|
}
|
@ -14,16 +14,10 @@ target_link_libraries(game PUBLIC
|
|||||||
stb
|
stb
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
# Set up flags
|
# Set up flags
|
||||||
target_compile_definitions(game PRIVATE
|
add_compile_definitions(
|
||||||
SETTING_PLATFORM_GLFW=1
|
SETTING_PLATFORM_GLFW=1
|
||||||
SETTING_PLATFORM=1
|
SETTING_PLATFORM=1
|
||||||
SETTING_PLATFORM_USE_GLAD=1
|
SETTING_PLATFORM_USE_GLAD=1
|
||||||
SETTING_ASSET_PREFIX="../../../assets/"
|
SETTING_ASSET_PREFIX="../../../assets/"
|
||||||
SETTING_GAME_NAME="DawnGame"
|
|
||||||
SETTING_GAME_POKER=1
|
|
||||||
SETTING_GAME_DAWN=2
|
|
||||||
SETTING_GAME_SANDBOX=3
|
|
||||||
SETTING_GAME=3
|
|
||||||
)
|
)
|
@ -10,6 +10,8 @@
|
|||||||
void engineInit(engine_t *engine) {
|
void engineInit(engine_t *engine) {
|
||||||
randSeed(123);
|
randSeed(123);
|
||||||
|
|
||||||
|
engine->name = SETTING_GAME_NAME;
|
||||||
|
|
||||||
epochInit(&engine->time);
|
epochInit(&engine->time);
|
||||||
renderInit();
|
renderInit();
|
||||||
inputInit(&engine->input);
|
inputInit(&engine->input);
|
||||||
|
@ -34,6 +34,7 @@ char * assetStringLoad(char *assetName) {
|
|||||||
|
|
||||||
assetbuffer_t * assetBufferOpen(char *assetName) {
|
assetbuffer_t * assetBufferOpen(char *assetName) {
|
||||||
// Get the directory based on the raw input by creating a new string.
|
// Get the directory based on the raw input by creating a new string.
|
||||||
|
FILE *fptr;
|
||||||
size_t lenAsset = strlen(assetName);// Get the length of asset
|
size_t lenAsset = strlen(assetName);// Get the length of asset
|
||||||
size_t lenPrefix = strlen(SETTING_ASSET_PREFIX);// Get the length of the prefix
|
size_t lenPrefix = strlen(SETTING_ASSET_PREFIX);// Get the length of the prefix
|
||||||
|
|
||||||
@ -48,7 +49,7 @@ assetbuffer_t * assetBufferOpen(char *assetName) {
|
|||||||
printf("Opening up %s\n", joined);
|
printf("Opening up %s\n", joined);
|
||||||
|
|
||||||
// Open the file pointer now.
|
// Open the file pointer now.
|
||||||
FILE *fptr = fopen(joined, "rb");
|
fptr = fopen(joined, "rb");
|
||||||
free(joined);// Free the string we just created
|
free(joined);// Free the string we just created
|
||||||
if(!fptr) return NULL;// File available?
|
if(!fptr) return NULL;// File available?
|
||||||
return (assetbuffer_t *)fptr;
|
return (assetbuffer_t *)fptr;
|
||||||
|
@ -4,36 +4,24 @@
|
|||||||
// https://opensource.org/licenses/MIT
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
#if !defined(GAME_TYPE)
|
||||||
|
#error You need to define the GAME_TYPE struct
|
||||||
|
#elif !defined(GAME_INIT)
|
||||||
|
#error You need to define the GAME_INIT method
|
||||||
|
#elif !defined(GAME_UPDATE)
|
||||||
|
#error You need to define the GAME_UPDATE method
|
||||||
|
#elif !defined(GAME_DISPOSE)
|
||||||
|
#error You need to define the GAME_DISPOSE method
|
||||||
|
#elif !defined(GAME_FILE)
|
||||||
|
#error You need to define the GAME_FILE string
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "../libs.h"
|
#include "../libs.h"
|
||||||
#include "../engine/engine.h"
|
#include "../engine/engine.h"
|
||||||
#include "../locale/language.h"
|
#include "../locale/language.h"
|
||||||
|
#include GAME_FILE
|
||||||
|
|
||||||
#define SETTING_GAME_SANDBOX 3
|
typedef GAME_TYPE game_t;
|
||||||
#define SETTING_GAME SETTING_GAME_SANDBOX
|
|
||||||
|
|
||||||
/** Describes the current game */
|
|
||||||
#if SETTING_GAME == SETTING_GAME_POKER
|
|
||||||
#include "poker/game.h"
|
|
||||||
typedef pokergame_t game_t;
|
|
||||||
#define GAME_INIT pokerGameInit
|
|
||||||
#define GAME_UPDATE pokerGameUpdate
|
|
||||||
#define GAME_DISPOSE pokerGameDispose
|
|
||||||
|
|
||||||
#elif SETTING_GAME == SETTING_GAME_DAWN
|
|
||||||
#include "dawn/dawngame.h"
|
|
||||||
typedef dawngame_t game_t;
|
|
||||||
#define GAME_INIT dawnGameInit
|
|
||||||
#define GAME_UPDATE dawnGameUpdate
|
|
||||||
#define GAME_DISPOSE dawnGameDispose
|
|
||||||
|
|
||||||
#elif SETTING_GAME == SETTING_GAME_SANDBOX
|
|
||||||
#include "sandbox/sandboxscene.h"
|
|
||||||
typedef sandboxscene_t game_t;
|
|
||||||
#define GAME_INIT sandboxSceneInit
|
|
||||||
#define GAME_UPDATE sandboxSceneUpdate
|
|
||||||
#define GAME_DISPOSE sandboxSceneDispose
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initialize the game context.
|
* Initialize the game context.
|
||||||
|
@ -9,22 +9,22 @@
|
|||||||
bool pokerGameAssetsInit(pokergameassets_t *assets) {
|
bool pokerGameAssetsInit(pokergameassets_t *assets) {
|
||||||
// Load the game's shader
|
// Load the game's shader
|
||||||
assetShaderLoad(&assets->shader,
|
assetShaderLoad(&assets->shader,
|
||||||
"shaders/textured.vert", "shaders/textured.frag"
|
"shared/shaders/textured.vert", "shared/shaders/textured.frag"
|
||||||
);
|
);
|
||||||
|
|
||||||
// Load the game's font
|
// Load the game's font
|
||||||
assetFontLoad(&assets->font, "fonts/opensans/OpenSans-Bold.ttf");
|
assetFontLoad(&assets->font, "shared/fonts/opensans/OpenSans-Bold.ttf");
|
||||||
|
|
||||||
// Initialize the language buffer.
|
// Initialize the language buffer.
|
||||||
languageInit(&assets->language, "locale/language/en-US.csv");
|
languageInit(&assets->language, "locale/language/en-US.csv");
|
||||||
|
|
||||||
// Load the world textures.
|
// Load the world textures.
|
||||||
assetTextureLoad(&assets->testTexture, "test_texture.png");
|
assetTextureLoad(&assets->testTexture, "shared/test_texture.png");
|
||||||
assetTextureLoad(&assets->cardTexture, "cards_normal.png");
|
assetTextureLoad(&assets->cardTexture, "poker/cards_normal.png");
|
||||||
assetTextureLoad(&assets->roomTexture, "world/pub/pub_skywall.png");
|
assetTextureLoad(&assets->roomTexture, "poker/world/pub/pub_skywall.png");
|
||||||
|
|
||||||
// Load the character textures.
|
// Load the character textures.
|
||||||
assetTextureLoad(&assets->pennyTexture, "characters/penny/sprites/sheet.png");
|
assetTextureLoad(&assets->pennyTexture, "poker/characters/penny/sprites/sheet.png");
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
12
tools/utils/args.js
Normal file
12
tools/utils/args.js
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
const args = process.argv.splice(2).reduce((x,y) => {
|
||||||
|
const bits = y.split('=');
|
||||||
|
const name = bits[0].replace('--', '');
|
||||||
|
const val = bits[1];
|
||||||
|
|
||||||
|
x[name] = val;
|
||||||
|
return x;
|
||||||
|
}, {});
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
args
|
||||||
|
};
|
103
tools/utils/image.js
Normal file
103
tools/utils/image.js
Normal file
@ -0,0 +1,103 @@
|
|||||||
|
const { PNG } = require("pngjs");
|
||||||
|
const path = require('path');
|
||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads an image into memory.
|
||||||
|
* @param image Image to load
|
||||||
|
* @returns A promise that resolves to the loaded image.
|
||||||
|
*/
|
||||||
|
const imageLoad = (image) => new Promise(resolve => {
|
||||||
|
fs.createReadStream(image)
|
||||||
|
.pipe(new PNG({ filterType: 4 }))
|
||||||
|
.on("parsed", function () {
|
||||||
|
// Normalize
|
||||||
|
const pixels = [];
|
||||||
|
for(let y = 0; y < this.height; y++) {
|
||||||
|
for(let x = 0; x < this.width; x++) {
|
||||||
|
const idx = (this.width * y + x) << 2;
|
||||||
|
const r = this.data[idx];
|
||||||
|
const g = this.data[idx + 1];
|
||||||
|
const b = this.data[idx + 2];
|
||||||
|
const a = this.data[idx + 3];
|
||||||
|
|
||||||
|
pixels.push({ r, g, b, a });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
resolve({ pixels, width: this.width, height: this.height });
|
||||||
|
})
|
||||||
|
;
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes an image to an output file.
|
||||||
|
* @param image Image to write.
|
||||||
|
* @param file File to write to.
|
||||||
|
* @returns A promise that, when resolved, has saved the image.
|
||||||
|
*/
|
||||||
|
const imageWrite = (image, file) => new Promise(resolve => {
|
||||||
|
const png = new PNG({ width: image.width, height: image.height });
|
||||||
|
png.width = image.width;
|
||||||
|
png.height = image.height;
|
||||||
|
|
||||||
|
for(let y = 0; y < image.height; y++) {
|
||||||
|
for(let x = 0; x < image.width; x++) {
|
||||||
|
const i = (image.width * y + x);
|
||||||
|
const idx = i << 2;
|
||||||
|
|
||||||
|
const pixel = image.pixels[i];
|
||||||
|
png.data[idx] = pixel.r;
|
||||||
|
png.data[idx + 1] = pixel.g;
|
||||||
|
png.data[idx + 2] = pixel.b;
|
||||||
|
png.data[idx + 3] = pixel.a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
png.pack().pipe(fs.createWriteStream(file))
|
||||||
|
.on('close', () => resolve(true))
|
||||||
|
;
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a blank image
|
||||||
|
* @param width Width of the image.
|
||||||
|
* @param height Height of the image.
|
||||||
|
* @param fill Optional pixel to fill with, defaults to 0,0,0,0
|
||||||
|
* @returns The newly created image.
|
||||||
|
*/
|
||||||
|
const imageCreate = (width, height, pixel) => {
|
||||||
|
if(!pixel || !pixel.r) pixel = { r:0, g:0, b:0, a:0 };
|
||||||
|
const pixels = [];
|
||||||
|
for(let i = 0; i < width * height; i++) pixels.push({ ...pixel });
|
||||||
|
return { pixels, width, height };
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies an area of a source image into a target image.
|
||||||
|
* @param target Target image to copy into.
|
||||||
|
* @param source Source image to copy from.
|
||||||
|
* @param tx Target X position to copy into
|
||||||
|
* @param ty Target Y position to copy into
|
||||||
|
* @param sub Optional source area to use, defined as { x, y, width, height }.
|
||||||
|
*/
|
||||||
|
const imageCopy = (target, source, tx, ty, sub) => {
|
||||||
|
if(!sub) sub = { x: 0, y: 0, width: source.width, height: source.height };
|
||||||
|
|
||||||
|
for(let x = sub.x; x < sub.x+sub.width; x++) {
|
||||||
|
for(let y = sub.y; y < sub.y+sub.height; y++) {
|
||||||
|
let absX = x - sub.x + tx;
|
||||||
|
let absY = y - sub.y + ty;
|
||||||
|
if(absX > target.width || absY > target.height) continue;
|
||||||
|
let ti = absY * target.width + absX;
|
||||||
|
let si = y * source.width + x;
|
||||||
|
target.pixels[ti] = { ...source.pixels[si] };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
imageWrite,
|
||||||
|
imageCreate,
|
||||||
|
imageLoad,
|
||||||
|
imageCopy
|
||||||
|
}
|
88
tools/vn/character-sheet-generator.js
Normal file
88
tools/vn/character-sheet-generator.js
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
const path = require('path');
|
||||||
|
const { imageCreate, imageWrite, imageLoad, imageCopy } = require('./../utils/image');
|
||||||
|
const fs = require('fs');
|
||||||
|
const xml = require('xml-js');
|
||||||
|
const { args } = require('./../utils/args');
|
||||||
|
|
||||||
|
// Parse Args
|
||||||
|
if(!args.in) throw new Error(`Missing in argument`);
|
||||||
|
if(!args.out) throw new Error(`Missing out argument`);
|
||||||
|
if(!args.in.endsWith('xml')) throw new Error(`Invalid in XML`);
|
||||||
|
if(!args.out.endsWith('png')) throw new Error(`Invalid out PNG`);
|
||||||
|
|
||||||
|
// Determine in and out.
|
||||||
|
const file = path.resolve(args.in);
|
||||||
|
const outFile = path.resolve(args.out);
|
||||||
|
if(fs.existsSync(outFile)) return;
|
||||||
|
|
||||||
|
// Load XML
|
||||||
|
const data = xml.xml2js(fs.readFileSync(file, 'utf-8'));
|
||||||
|
const [ character ] = data.elements;
|
||||||
|
|
||||||
|
// Validate file.
|
||||||
|
if(!character.attributes.context) throw new Error(`Missing context`)
|
||||||
|
const dir = path.resolve('.', 'assets', character.attributes.context);
|
||||||
|
|
||||||
|
// Parse base and layers
|
||||||
|
const base = character.elements.find(e => e.name == 'base').attributes;
|
||||||
|
if(!base) throw new Error(`Failed to find base`);
|
||||||
|
const layers = character.elements
|
||||||
|
.filter(e => e.name == 'layer')
|
||||||
|
.map(e => e.attributes)
|
||||||
|
.map(e => ({
|
||||||
|
...e,
|
||||||
|
x: parseInt(e.x),
|
||||||
|
y: parseInt(e.y),
|
||||||
|
width: parseInt(e.width),
|
||||||
|
height: parseInt(e.height)
|
||||||
|
}))
|
||||||
|
;
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
// Load the base
|
||||||
|
const baseImage = await imageLoad(path.join(dir, base.file));
|
||||||
|
|
||||||
|
let columnsMax = 0;
|
||||||
|
let widthMax = 0;
|
||||||
|
layers.forEach((layer,row) => {
|
||||||
|
if(!layer.width || !layer.height || !layer.x || !layer.y) {
|
||||||
|
throw new Error(`Missing layer info`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const layerDir = path.join(dir, layer.directory);
|
||||||
|
const scan = fs.readdirSync(layerDir);
|
||||||
|
columnsMax = Math.max(scan.length, columnsMax);
|
||||||
|
widthMax = Math.max(widthMax, layer.width);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create the output buffer
|
||||||
|
const out = imageCreate(
|
||||||
|
baseImage.width + (columnsMax * widthMax),
|
||||||
|
baseImage.height
|
||||||
|
);
|
||||||
|
|
||||||
|
// Copy the base
|
||||||
|
imageCopy(out, baseImage, 0, 0);
|
||||||
|
|
||||||
|
// Now begin copying the children, row is defined by the directory
|
||||||
|
let y = 0;
|
||||||
|
for(let row = 0; row < layers.length; row++) {
|
||||||
|
const layer = layers[row];
|
||||||
|
const layerDir = path.join(dir, layer.directory);
|
||||||
|
const scan = fs.readdirSync(layerDir);
|
||||||
|
|
||||||
|
// Column defined by the file index
|
||||||
|
for(let col = 0; col < scan.length; col++) {
|
||||||
|
const img = await imageLoad(path.join(layerDir, scan[col]));
|
||||||
|
console.log('Copying', scan[col]);
|
||||||
|
imageCopy(out, img,
|
||||||
|
baseImage.width+(col*layer.width), y,
|
||||||
|
layer
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
y += layer.height;
|
||||||
|
}
|
||||||
|
|
||||||
|
await imageWrite(out, outFile);
|
||||||
|
})().catch(console.error);
|
Reference in New Issue
Block a user