Added language support to main menu and sign
This commit is contained in:
@ -1,7 +1,16 @@
|
|||||||
{
|
{
|
||||||
"general": {
|
"main_menu": {
|
||||||
"test": {
|
"new_game": "New Game",
|
||||||
"test": "test"
|
"load_game": "Load Game",
|
||||||
|
"options": "Options",
|
||||||
|
"exit": "Exit"
|
||||||
|
},
|
||||||
|
"maps": {
|
||||||
|
"testmap": {
|
||||||
|
"sign": {
|
||||||
|
"text": "This is a sign.",
|
||||||
|
"name": "Sign"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -40,7 +40,7 @@
|
|||||||
"type": 3,
|
"type": 3,
|
||||||
"x": 3,
|
"x": 3,
|
||||||
"y": 8,
|
"y": 8,
|
||||||
"text": "This is a sign."
|
"text": "maps.testmap.sign"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": 4,
|
"type": 4,
|
||||||
|
@ -28,28 +28,17 @@ game_t GAME;
|
|||||||
|
|
||||||
void gameInit() {
|
void gameInit() {
|
||||||
memset(&GAME, 0, sizeof(game_t));
|
memset(&GAME, 0, sizeof(game_t));
|
||||||
|
|
||||||
timeInit();
|
timeInit();
|
||||||
inputInit();
|
inputInit();
|
||||||
displayInit();
|
displayInit();
|
||||||
assetInit();
|
assetInit();
|
||||||
|
assetLanguageLoad("en.json");
|
||||||
|
|
||||||
textboxInit();
|
textboxInit();
|
||||||
testMenuInit();
|
testMenuInit();
|
||||||
mainMenuInit();
|
mainMenuInit();
|
||||||
|
conversationInit();
|
||||||
assetLanguageLoad("en.json");
|
|
||||||
|
|
||||||
char_t buffer[LANGUAGE_STRING_LENGTH_MAX];
|
|
||||||
languageGet(buffer, "general.test.test");
|
|
||||||
|
|
||||||
conversation_t conversation;
|
|
||||||
conversationInit(
|
|
||||||
&conversation,
|
|
||||||
conversationTestInit,
|
|
||||||
conversationTestUpdate
|
|
||||||
);
|
|
||||||
conversationUpdate(&conversation);
|
|
||||||
|
|
||||||
GAME.state = GAME_STATE_INITIAL;
|
GAME.state = GAME_STATE_INITIAL;
|
||||||
}
|
}
|
||||||
@ -57,6 +46,7 @@ void gameInit() {
|
|||||||
gameupdateresult_t gameUpdate(const float_t delta) {
|
gameupdateresult_t gameUpdate(const float_t delta) {
|
||||||
timeUpdate(delta);
|
timeUpdate(delta);
|
||||||
inputUpdate();
|
inputUpdate();
|
||||||
|
conversationUpdate();
|
||||||
|
|
||||||
switch(GAME.state) {
|
switch(GAME.state) {
|
||||||
case GAME_STATE_INITIAL:
|
case GAME_STATE_INITIAL:
|
||||||
|
@ -14,23 +14,29 @@ void languageInit() {
|
|||||||
memset(&LANGUAGE, 0, sizeof(language_t));
|
memset(&LANGUAGE, 0, sizeof(language_t));
|
||||||
}
|
}
|
||||||
|
|
||||||
int32_t languageGet(
|
const char_t * languageGetPointer(const char_t *key) {
|
||||||
char_t *buffer,
|
|
||||||
const char_t *key,
|
|
||||||
...
|
|
||||||
) {
|
|
||||||
assertNotNull(key, "Key cannot be NULL.");
|
assertNotNull(key, "Key cannot be NULL.");
|
||||||
|
|
||||||
int32_t i;
|
language_t *lang = &LANGUAGE;
|
||||||
|
lang->count;
|
||||||
|
|
||||||
|
int32_t i = 0;
|
||||||
while(i < LANGUAGE.count) {
|
while(i < LANGUAGE.count) {
|
||||||
if(strcmp(key, LANGUAGE.keys[i]) != 0) {
|
if(strcmp(key, LANGUAGE.keys[i]) != 0) {
|
||||||
i++;
|
i++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
return LANGUAGE.strings[i];
|
||||||
if(buffer != NULL) strcpy(buffer, LANGUAGE.strings[i]);
|
|
||||||
return strlen(LANGUAGE.strings[i]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return -1;
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t languageGet(char_t *buffer, const char_t *key) {
|
||||||
|
const char_t *str = languageGetPointer(key);
|
||||||
|
if(str == NULL) return -1;
|
||||||
|
if(buffer == NULL) return strlen(str);
|
||||||
|
|
||||||
|
strcpy(buffer, str);
|
||||||
|
return strlen(str);
|
||||||
}
|
}
|
@ -25,16 +25,20 @@ extern language_t LANGUAGE;
|
|||||||
*/
|
*/
|
||||||
void languageInit();
|
void languageInit();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the pointer to the language string for the given key. Pointer should not
|
||||||
|
* be modified or freed.
|
||||||
|
*
|
||||||
|
* @param key The key to get the string for.
|
||||||
|
* @return The pointer to the string.
|
||||||
|
*/
|
||||||
|
const char_t * languageGetPointer(const char_t *key);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the language string for the given key.
|
* Returns the language string for the given key.
|
||||||
*
|
*
|
||||||
* @param buffer The buffer to write the string to, or NULL to get length.
|
* @param buffer The buffer to write the string to, or NULL to get length.
|
||||||
* @param key The key to get the string for.
|
* @param key The key to get the string for.
|
||||||
* @param ... The arguments to replace in the string.
|
|
||||||
* @return The length of the string.
|
* @return The length of the string.
|
||||||
*/
|
*/
|
||||||
int32_t languageGet(
|
int32_t languageGet(char_t *buffer, const char_t *key);
|
||||||
char_t *buffer,
|
|
||||||
const char_t *key,
|
|
||||||
...
|
|
||||||
);
|
|
@ -9,5 +9,6 @@
|
|||||||
target_sources(${DAWN_TARGET_NAME}
|
target_sources(${DAWN_TARGET_NAME}
|
||||||
PRIVATE
|
PRIVATE
|
||||||
conversation.c
|
conversation.c
|
||||||
|
conversationsign.c
|
||||||
testconversation.c
|
testconversation.c
|
||||||
)
|
)
|
@ -8,44 +8,48 @@
|
|||||||
#include "conversation.h"
|
#include "conversation.h"
|
||||||
#include "assert/assert.h"
|
#include "assert/assert.h"
|
||||||
|
|
||||||
void conversationInit(
|
conversation_t CONVERSATION;
|
||||||
conversation_t *conversation,
|
|
||||||
const conversationcallback_t init,
|
|
||||||
const conversationcallback_t update
|
|
||||||
) {
|
|
||||||
assertNotNull(conversation, "Conversation is NULL.");
|
|
||||||
|
|
||||||
memset(conversation, 0, sizeof(conversation_t));
|
void conversationInit() {
|
||||||
conversation->init = init;
|
memset(&CONVERSATION, 0, sizeof(conversation_t));
|
||||||
conversation->update = update;
|
|
||||||
conversation->index = CONVERSATION_NOT_INITIALIZED;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void conversationUpdate(conversation_t *conversation) {
|
void conversationSet(
|
||||||
|
const conversationcallback_t init,
|
||||||
|
const conversationcallback_t update,
|
||||||
|
void *user
|
||||||
|
) {
|
||||||
|
CONVERSATION.init = init;
|
||||||
|
CONVERSATION.update = update;
|
||||||
|
CONVERSATION.index = CONVERSATION_NOT_INITIALIZED;
|
||||||
|
CONVERSATION.user = user;
|
||||||
|
}
|
||||||
|
|
||||||
|
void conversationUpdate() {
|
||||||
|
if(CONVERSATION.init == NULL || CONVERSATION.update == NULL) return;
|
||||||
|
|
||||||
uint16_t ret;
|
uint16_t ret;
|
||||||
|
|
||||||
assertNotNull(conversation, "Conversation is NULL.");
|
|
||||||
|
|
||||||
// Init or update.
|
// Init or update.
|
||||||
switch(conversation->index) {
|
switch(CONVERSATION.index) {
|
||||||
case CONVERSATION_NOT_INITIALIZED:
|
case CONVERSATION_NOT_INITIALIZED:
|
||||||
conversation->index = 0;
|
CONVERSATION.index = 0;
|
||||||
ret = conversation->init(
|
ret = CONVERSATION.init(
|
||||||
conversation, conversation->index
|
&CONVERSATION, CONVERSATION.index
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
ret = conversation->update(conversation, conversation->index);
|
ret = CONVERSATION.update(&CONVERSATION, CONVERSATION.index);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check ret value and update conversation.
|
// Check ret value and update conversation.
|
||||||
switch(ret) {
|
switch(ret) {
|
||||||
case CONVERSATION_NEXT:
|
case CONVERSATION_NEXT:
|
||||||
conversation->index++;
|
CONVERSATION.index++;
|
||||||
conversation->index = conversation->init(
|
CONVERSATION.index = CONVERSATION.init(
|
||||||
conversation, conversation->index
|
&CONVERSATION, CONVERSATION.index
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -53,8 +57,8 @@ void conversationUpdate(conversation_t *conversation) {
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
case CONVERSATION_RESTART:
|
case CONVERSATION_RESTART:
|
||||||
conversation->index = 0;
|
CONVERSATION.index = 0;
|
||||||
conversation->init(conversation, conversation->index);
|
CONVERSATION.init(&CONVERSATION, CONVERSATION.index);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case CONVERSATION_INVALID:
|
case CONVERSATION_INVALID:
|
||||||
@ -62,13 +66,12 @@ void conversationUpdate(conversation_t *conversation) {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case CONVERSATION_DONE:
|
case CONVERSATION_DONE:
|
||||||
|
CONVERSATION.init = NULL;
|
||||||
|
CONVERSATION.update = NULL;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
conversation->index = ret;
|
CONVERSATION.index = ret;
|
||||||
conversation->index = conversation->init(
|
|
||||||
conversation, conversation->index
|
|
||||||
);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -28,21 +28,27 @@ typedef struct _conversation_t {
|
|||||||
void *user;
|
void *user;
|
||||||
} conversation_t;
|
} conversation_t;
|
||||||
|
|
||||||
|
extern conversation_t CONVERSATION;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initializes a conversation object.
|
* Initializes the conversation object.
|
||||||
*
|
|
||||||
* @param conversation Conversation to initialize.
|
|
||||||
* @param update Function to handle conversation updates.
|
|
||||||
*/
|
*/
|
||||||
void conversationInit(
|
void conversationInit();
|
||||||
conversation_t *conversation,
|
|
||||||
|
/**
|
||||||
|
* Sets the conversation object.
|
||||||
|
*
|
||||||
|
* @param init The initialization callback.
|
||||||
|
* @param update The update callback.
|
||||||
|
* @param user The user data.
|
||||||
|
*/
|
||||||
|
void conversationSet(
|
||||||
const conversationcallback_t init,
|
const conversationcallback_t init,
|
||||||
const conversationcallback_t update
|
const conversationcallback_t update,
|
||||||
|
void *user
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update a given conversation.
|
* Update a given conversation.
|
||||||
*
|
|
||||||
* @param conversastion Conversation to update.
|
|
||||||
*/
|
*/
|
||||||
void conversationUpdate(conversation_t *conversation);
|
void conversationUpdate();
|
32
src/dawn/rpg/conversation/conversationsign.c
Normal file
32
src/dawn/rpg/conversation/conversationsign.c
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2024 Dominic Masters
|
||||||
|
*
|
||||||
|
* This software is released under the MIT License.
|
||||||
|
* https://opensource.org/licenses/MIT
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "conversationsign.h"
|
||||||
|
#include "assert/assert.h"
|
||||||
|
#include "rpg/entity/entity.h"
|
||||||
|
#include "locale/language.h"
|
||||||
|
|
||||||
|
uint16_t conversationSignInit(conversation_t *convo, const uint16_t i) {
|
||||||
|
assertNotNull(convo, "Conversation is NULL!");
|
||||||
|
assertNotNull(convo->user, "Conversation user is NULL!");
|
||||||
|
|
||||||
|
entity_t *e = (entity_t*)convo->user;
|
||||||
|
assertTrue(e->type == ENTITY_TYPE_SIGN, "Entity is not a sign!");
|
||||||
|
textboxSetText("maps.testmap.sign.name", "maps.testmap.sign.text");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t conversationSignUpdate(conversation_t *convo, const uint16_t i) {
|
||||||
|
switch(i) {
|
||||||
|
case 0:
|
||||||
|
return textboxIsOpen() ? CONVERSATION_CONTINUE : CONVERSATION_DONE;
|
||||||
|
|
||||||
|
default:
|
||||||
|
return CONVERSATION_INVALID;
|
||||||
|
}
|
||||||
|
}
|
12
src/dawn/rpg/conversation/conversationsign.h
Normal file
12
src/dawn/rpg/conversation/conversationsign.h
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2024 Dominic Masters
|
||||||
|
*
|
||||||
|
* This software is released under the MIT License.
|
||||||
|
* https://opensource.org/licenses/MIT
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#include "conversation.h"
|
||||||
|
|
||||||
|
uint16_t conversationSignInit(conversation_t *convo, const uint16_t i);
|
||||||
|
uint16_t conversationSignUpdate(conversation_t *convo, const uint16_t i);
|
@ -8,6 +8,9 @@
|
|||||||
#include "interact.h"
|
#include "interact.h"
|
||||||
#include "assert/assert.h"
|
#include "assert/assert.h"
|
||||||
#include "game/game.h"
|
#include "game/game.h"
|
||||||
|
#include "locale/language.h"
|
||||||
|
|
||||||
|
#include "rpg/conversation/conversationsign.h"
|
||||||
|
|
||||||
void entityInteractEntity(
|
void entityInteractEntity(
|
||||||
entity_t *source,
|
entity_t *source,
|
||||||
@ -29,7 +32,7 @@ void entityInteractEntity(
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
case ENTITY_TYPE_SIGN:
|
case ENTITY_TYPE_SIGN:
|
||||||
textboxSetText("Sign", target->sign.text);
|
conversationSet(conversationSignInit, conversationSignUpdate, target);
|
||||||
return;
|
return;
|
||||||
|
|
||||||
case ENTITY_TYPE_DOOR:
|
case ENTITY_TYPE_DOOR:
|
||||||
|
@ -7,15 +7,16 @@
|
|||||||
|
|
||||||
#include "mainmenu.h"
|
#include "mainmenu.h"
|
||||||
#include "game/game.h"
|
#include "game/game.h"
|
||||||
|
#include "locale/language.h"
|
||||||
|
|
||||||
menu_t MAIN_MENU;
|
menu_t MAIN_MENU;
|
||||||
|
|
||||||
void mainMenuInit() {
|
void mainMenuInit() {
|
||||||
const char_t* strings[] = {
|
const char_t* strings[] = {
|
||||||
"New Game",
|
languageGetPointer("main_menu.new_game"),
|
||||||
"Load Game",
|
languageGetPointer("main_menu.load_game"),
|
||||||
"Options",
|
languageGetPointer("main_menu.options"),
|
||||||
"Exit"
|
languageGetPointer("main_menu.exit")
|
||||||
};
|
};
|
||||||
|
|
||||||
uint16_t columns = 1;
|
uint16_t columns = 1;
|
||||||
@ -35,21 +36,25 @@ void mainMenuSelectCallback(
|
|||||||
const uint16_t y,
|
const uint16_t y,
|
||||||
const char_t *str
|
const char_t *str
|
||||||
) {
|
) {
|
||||||
if(strcmp(str, "New Game") == 0) {
|
// New Game
|
||||||
|
if(y == 0) {
|
||||||
GAME.mapNext = MAP_LIST_TEST;
|
GAME.mapNext = MAP_LIST_TEST;
|
||||||
GAME.state = GAME_STATE_MAP_CHANGE;
|
GAME.state = GAME_STATE_MAP_CHANGE;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(strcmp(str, "Load Game") == 0) {
|
// Load Game
|
||||||
|
if(y == 1) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(strcmp(str, "Options") == 0) {
|
// Options
|
||||||
|
if(y == 2) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(strcmp(str, "Exit") == 0) {
|
// Exit
|
||||||
|
if(y == 3) {
|
||||||
GAME.shouldExit = true;
|
GAME.shouldExit = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
#include "assert/assert.h"
|
#include "assert/assert.h"
|
||||||
#include "input.h"
|
#include "input.h"
|
||||||
#include "game/time.h"
|
#include "game/time.h"
|
||||||
|
#include "locale/language.h"
|
||||||
|
|
||||||
textbox_t TEXTBOX;
|
textbox_t TEXTBOX;
|
||||||
|
|
||||||
@ -20,18 +21,23 @@ void textboxSetText(
|
|||||||
const char_t *title,
|
const char_t *title,
|
||||||
const char_t *text
|
const char_t *text
|
||||||
) {
|
) {
|
||||||
|
size_t len;
|
||||||
|
const char_t *temp;
|
||||||
assertNotNull(text, "Text cannot be NULL.");
|
assertNotNull(text, "Text cannot be NULL.");
|
||||||
|
|
||||||
// Setup text copies
|
// Copy translation
|
||||||
size_t len = strlen(text);
|
temp = languageGetPointer(text);
|
||||||
|
len = strlen(temp);
|
||||||
assertTrue(len < TEXTBOX_TEXT_MAX, "Text is too long.");
|
assertTrue(len < TEXTBOX_TEXT_MAX, "Text is too long.");
|
||||||
strcpy(TEXTBOX.text, text);
|
strcpy(TEXTBOX.text, temp);
|
||||||
TEXTBOX.textLength = len;
|
TEXTBOX.textLength = len;
|
||||||
|
|
||||||
// Setup title
|
// Setup title
|
||||||
if(title) {
|
if(title) {
|
||||||
assertTrue(strlen(title) < TEXTBOX_TITLE_MAX, "Title is too long.");
|
temp = languageGetPointer(title);
|
||||||
strcpy(TEXTBOX.title, title);
|
len = strlen(temp);
|
||||||
|
assertTrue(len < TEXTBOX_TITLE_MAX, "Title is too long.");
|
||||||
|
strcpy(TEXTBOX.title, temp);
|
||||||
} else {
|
} else {
|
||||||
TEXTBOX.title[0] = '\0';
|
TEXTBOX.title[0] = '\0';
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user