TIC-80-guile/src/config.c

364 lines
7.5 KiB
C

// MIT License
// Copyright (c) 2017 Vadim Grigoruk @nesbox // grigoruk@gmail.com
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
#include "config.h"
#include "fs.h"
#include <lua.h>
#include <lauxlib.h>
#include <lualib.h>
static void readConfigVideoLength(Config* config, lua_State* lua)
{
lua_getglobal(lua, "GIF_LENGTH");
if(lua_isinteger(lua, -1))
config->data.gifLength = (s32)lua_tointeger(lua, -1);
lua_pop(lua, 1);
}
static void readConfigVideoScale(Config* config, lua_State* lua)
{
lua_getglobal(lua, "GIF_SCALE");
if(lua_isinteger(lua, -1))
config->data.gifScale = (s32)lua_tointeger(lua, -1);
lua_pop(lua, 1);
}
static void readConfigCheckNewVersion(Config* config, lua_State* lua)
{
lua_getglobal(lua, "CHECK_NEW_VERSION");
if(lua_isboolean(lua, -1))
config->data.checkNewVersion = lua_toboolean(lua, -1);
lua_pop(lua, 1);
}
static void readConfigNoSound(Config* config, lua_State* lua)
{
lua_getglobal(lua, "NO_SOUND");
if(lua_isboolean(lua, -1))
config->data.noSound = lua_toboolean(lua, -1);
lua_pop(lua, 1);
}
static void readConfigShowSync(Config* config, lua_State* lua)
{
lua_getglobal(lua, "SHOW_SYNC");
if(lua_isboolean(lua, -1))
config->data.showSync = lua_toboolean(lua, -1);
lua_pop(lua, 1);
}
static void readConfigCrtMonitor(Config* config, lua_State* lua)
{
lua_getglobal(lua, "CRT_MONITOR");
if(lua_isboolean(lua, -1))
config->data.crtMonitor = lua_toboolean(lua, -1);
lua_pop(lua, 1);
}
static void readConfigUiScale(Config* config, lua_State* lua)
{
lua_getglobal(lua, "UI_SCALE");
if(lua_isinteger(lua, -1))
config->data.uiScale = lua_tointeger(lua, -1);
lua_pop(lua, 1);
}
static void readConfigCrtShader(Config* config, lua_State* lua)
{
lua_getglobal(lua, "CRT_SHADER");
if(lua_isstring(lua, -1))
{
if(!config->data.crtShader)
config->data.crtShader = calloc(1, sizeof(tic_code));
strcpy((char*)config->data.crtShader, lua_tostring(lua, -1));
}
lua_pop(lua, 1);
}
static void readCursorTheme(Config* config, lua_State* lua)
{
lua_getfield(lua, -1, "CURSOR");
if(lua_type(lua, -1) == LUA_TTABLE)
{
{
lua_getfield(lua, -1, "ARROW");
if(lua_isinteger(lua, -1))
{
config->data.theme.cursor.arrow = (s32)lua_tointeger(lua, -1);
}
lua_pop(lua, 1);
}
{
lua_getfield(lua, -1, "HAND");
if(lua_isinteger(lua, -1))
{
config->data.theme.cursor.hand = (s32)lua_tointeger(lua, -1);
}
lua_pop(lua, 1);
}
{
lua_getfield(lua, -1, "IBEAM");
if(lua_isinteger(lua, -1))
{
config->data.theme.cursor.ibeam = (s32)lua_tointeger(lua, -1);
}
lua_pop(lua, 1);
}
{
lua_getfield(lua, -1, "PIXEL_PERFECT");
if(lua_isboolean(lua, -1))
config->data.theme.cursor.pixelPerfect = lua_toboolean(lua, -1);
lua_pop(lua, 1);
}
}
lua_pop(lua, 1);
}
static void readCodeTheme(Config* config, lua_State* lua)
{
lua_getfield(lua, -1, "CODE");
if(lua_type(lua, -1) == LUA_TTABLE)
{
static const char* Syntax[] = {"STRING", "NUMBER", "KEYWORD", "API", "COMMENT", "SIGN", "VAR", "OTHER"};
for(s32 i = 0; i < COUNT_OF(Syntax); i++)
{
lua_getfield(lua, -1, Syntax[i]);
if(lua_isinteger(lua, -1))
((u8*)&config->data.theme.code.syntax)[i] = (u8)lua_tointeger(lua, -1);
lua_pop(lua, 1);
}
static const char* Fields[] = {"BG", "SELECT", "CURSOR"};
for(s32 i = 0; i < COUNT_OF(Fields); i++)
{
lua_getfield(lua, -1, Fields[i]);
if(lua_isinteger(lua, -1))
((u8*)&config->data.theme.code.bg)[i] = (u8)lua_tointeger(lua, -1);
lua_pop(lua, 1);
}
{
lua_getfield(lua, -1, "SHADOW");
if(lua_isboolean(lua, -1))
config->data.theme.code.shadow = lua_toboolean(lua, -1);
lua_pop(lua, 1);
}
{
lua_getfield(lua, -1, "ALT_FONT");
if(lua_isboolean(lua, -1))
config->data.theme.code.altFont = lua_toboolean(lua, -1);
lua_pop(lua, 1);
}
}
lua_pop(lua, 1);
}
static void readGamepadTheme(Config* config, lua_State* lua)
{
lua_getfield(lua, -1, "GAMEPAD");
if(lua_type(lua, -1) == LUA_TTABLE)
{
lua_getfield(lua, -1, "TOUCH");
if(lua_type(lua, -1) == LUA_TTABLE)
{
lua_getfield(lua, -1, "ALPHA");
if(lua_isinteger(lua, -1))
config->data.theme.gamepad.touch.alpha = (u8)lua_tointeger(lua, -1);
lua_pop(lua, 1);
}
lua_pop(lua, 1);
}
lua_pop(lua, 1);
}
static void readTheme(Config* config, lua_State* lua)
{
lua_getglobal(lua, "THEME");
if(lua_type(lua, -1) == LUA_TTABLE)
{
readCursorTheme(config, lua);
readCodeTheme(config, lua);
readGamepadTheme(config, lua);
}
lua_pop(lua, 1);
}
static void readConfig(Config* config)
{
lua_State* lua = luaL_newstate();
if(lua)
{
if(luaL_loadstring(lua, config->cart.code.data) == LUA_OK && lua_pcall(lua, 0, LUA_MULTRET, 0) == LUA_OK)
{
readConfigVideoLength(config, lua);
readConfigVideoScale(config, lua);
readConfigCheckNewVersion(config, lua);
readConfigNoSound(config, lua);
readConfigShowSync(config, lua);
readConfigCrtMonitor(config, lua);
readConfigUiScale(config, lua);
readTheme(config, lua);
readConfigCrtShader(config, lua);
}
lua_close(lua);
}
}
static void update(Config* config, const u8* buffer, size_t size)
{
config->tic->api.load(&config->cart, buffer, size, true);
readConfig(config);
studioConfigChanged();
}
static void setDefault(Config* config)
{
memset(&config->data, 0, sizeof(StudioConfig));
config->data.cart = &config->cart;
{
static const u8 DefaultBiosZip[] =
{
#include "../bin/assets/config.tic.dat"
};
u8* embedBios = NULL;
s32 size = unzip((u8**)&embedBios, DefaultBiosZip, sizeof DefaultBiosZip);
if(embedBios)
{
update(config, embedBios, size);
free(embedBios);
}
}
}
static void saveConfig(Config* config, bool overwrite)
{
u8* buffer = malloc(sizeof(tic_cartridge));
if(buffer)
{
s32 size = config->tic->api.save(&config->cart, buffer);
fsSaveRootFile(config->fs, CONFIG_TIC_PATH, buffer, size, overwrite);
free(buffer);
}
}
static void reset(Config* config)
{
setDefault(config);
saveConfig(config, true);
}
static void save(Config* config)
{
memcpy(&config->cart, &config->tic->cart, sizeof(tic_cartridge));
readConfig(config);
saveConfig(config, true);
studioConfigChanged();
}
void initConfig(Config* config, tic_mem* tic, FileSystem* fs)
{
{
config->tic = tic;
config->save = save;
config->reset = reset;
config->fs = fs;
}
setDefault(config);
s32 size = 0;
u8* data = (u8*)fsLoadRootFile(fs, CONFIG_TIC_PATH, &size);
if(data)
{
update(config, data, size);
free(data);
}
else saveConfig(config, false);
tic->api.reset(tic);
}