Textbox rendering back.
This commit is contained in:
@ -12,7 +12,7 @@ set(DAWN_CACHE_TARGET "dawn-target")
|
|||||||
set(DAWN_TARGET_NAME "Dawn")
|
set(DAWN_TARGET_NAME "Dawn")
|
||||||
|
|
||||||
if(NOT DEFINED DAWN_TARGET)
|
if(NOT DEFINED DAWN_TARGET)
|
||||||
set(DAWN_TARGET linux-x64-terminal CACHE INTERNAL ${DAWN_CACHE_TARGET})
|
set(DAWN_TARGET linux-x64-glfw CACHE INTERNAL ${DAWN_CACHE_TARGET})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Set Common Build Variables
|
# Set Common Build Variables
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
# Sources
|
# Sources
|
||||||
target_sources(${DAWN_TARGET_NAME}
|
target_sources(${DAWN_TARGET_NAME}
|
||||||
PRIVATE
|
PRIVATE
|
||||||
|
drawstateoverworld.c
|
||||||
drawmap.c
|
drawmap.c
|
||||||
drawtext.c
|
drawtext.c
|
||||||
drawshape.c
|
drawshape.c
|
||||||
|
39
src/dawn/display/draw/drawstateoverworld.c
Normal file
39
src/dawn/display/draw/drawstateoverworld.c
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2024 Dominic Masters
|
||||||
|
*
|
||||||
|
* This software is released under the MIT License.
|
||||||
|
* https://opensource.org/licenses/MIT
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "drawstateoverworld.h"
|
||||||
|
#include "ui/textbox.h"
|
||||||
|
#include "rpg/world/map.h"
|
||||||
|
#include "game/game.h"
|
||||||
|
#include "display/draw/drawmap.h"
|
||||||
|
#include "display/draw/drawui.h"
|
||||||
|
|
||||||
|
void drawStateOverworld() {
|
||||||
|
map_t *map = GAME.currentMap;
|
||||||
|
if(map == NULL) return;
|
||||||
|
|
||||||
|
// Try get player
|
||||||
|
entity_t *player = mapEntityGetByType(map, ENTITY_TYPE_PLAYER);
|
||||||
|
uint16_t cameraPositionX, cameraPositionY;
|
||||||
|
if(player == NULL) {
|
||||||
|
cameraPositionX = 0;
|
||||||
|
cameraPositionY = 0;
|
||||||
|
} else {
|
||||||
|
cameraPositionX = player->x;
|
||||||
|
cameraPositionY = player->y;
|
||||||
|
}
|
||||||
|
|
||||||
|
drawMap(
|
||||||
|
GAME.currentMap,
|
||||||
|
cameraPositionX, cameraPositionY,
|
||||||
|
0, 0,
|
||||||
|
FRAME_WIDTH, FRAME_HEIGHT
|
||||||
|
);
|
||||||
|
|
||||||
|
// Draw UI
|
||||||
|
drawUITextbox();
|
||||||
|
}
|
14
src/dawn/display/draw/drawstateoverworld.h
Normal file
14
src/dawn/display/draw/drawstateoverworld.h
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2024 Dominic Masters
|
||||||
|
*
|
||||||
|
* This software is released under the MIT License.
|
||||||
|
* https://opensource.org/licenses/MIT
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#include "display/frame.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Draws the overworld state.
|
||||||
|
*/
|
||||||
|
void drawStateOverworld();
|
@ -7,20 +7,22 @@
|
|||||||
|
|
||||||
#include "drawtext.h"
|
#include "drawtext.h"
|
||||||
#include "assert/assert.h"
|
#include "assert/assert.h"
|
||||||
|
#include "util/math.h"
|
||||||
|
|
||||||
void drawText(
|
void drawText(
|
||||||
const char_t *text,
|
const char_t *text,
|
||||||
|
const size_t len,
|
||||||
const uint16_t x,
|
const uint16_t x,
|
||||||
const uint16_t y,
|
const uint16_t y,
|
||||||
const uint8_t color
|
const uint8_t color
|
||||||
) {
|
) {
|
||||||
size_t len = strlen(text);
|
|
||||||
if(len == 0) return;
|
if(len == 0) return;
|
||||||
|
size_t rLen = mathMin(len, strlen(text));
|
||||||
|
|
||||||
assertTrue(x + len <= FRAME_WIDTH, "Text is too long.");
|
assertTrue(x + rLen <= FRAME_WIDTH, "Text is too long.");
|
||||||
assertTrue(y < FRAME_HEIGHT, "Text is too low.");
|
assertTrue(y < FRAME_HEIGHT, "Text is too low.");
|
||||||
|
|
||||||
uint16_t i = y * FRAME_WIDTH + x;
|
uint16_t i = y * FRAME_WIDTH + x;
|
||||||
memcpy(&FRAME_BUFFER[i], text, len);
|
memcpy(&FRAME_BUFFER[i], text, rLen);
|
||||||
memset(&FRAME_COLOR[i], color, len);
|
memset(&FRAME_COLOR[i], color, rLen);
|
||||||
}
|
}
|
@ -12,12 +12,14 @@
|
|||||||
* Draws some text to the frame buffer.
|
* Draws some text to the frame buffer.
|
||||||
*
|
*
|
||||||
* @param text The text to draw.
|
* @param text The text to draw.
|
||||||
|
* @param len The length of the text to draw.
|
||||||
* @param x The x position to draw the text.
|
* @param x The x position to draw the text.
|
||||||
* @param y The y position to draw the text.
|
* @param y The y position to draw the text.
|
||||||
* @param color The color to draw the text.
|
* @param color The color to draw the text.
|
||||||
*/
|
*/
|
||||||
void drawText(
|
void drawText(
|
||||||
const char_t *text,
|
const char_t *text,
|
||||||
|
const size_t len,
|
||||||
const uint16_t x,
|
const uint16_t x,
|
||||||
const uint16_t y,
|
const uint16_t y,
|
||||||
const uint8_t color
|
const uint8_t color
|
||||||
|
@ -8,6 +8,9 @@
|
|||||||
#include "drawui.h"
|
#include "drawui.h"
|
||||||
#include "drawshape.h"
|
#include "drawshape.h"
|
||||||
#include "assert/assert.h"
|
#include "assert/assert.h"
|
||||||
|
#include "ui/textbox.h"
|
||||||
|
#include "game/time.h"
|
||||||
|
#include "display/draw/drawtext.h"
|
||||||
|
|
||||||
void drawUIBox(
|
void drawUIBox(
|
||||||
const uint16_t x,
|
const uint16_t x,
|
||||||
@ -54,3 +57,39 @@ void drawUIBox(
|
|||||||
if(!fill) return;
|
if(!fill) return;
|
||||||
drawBox(x + 1, y + 1, width - 2, height - 2, ' ', COLOR_BLACK);
|
drawBox(x + 1, y + 1, width - 2, height - 2, ' ', COLOR_BLACK);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void drawUITextbox() {
|
||||||
|
if(!textboxIsOpen()) return;
|
||||||
|
|
||||||
|
// Border
|
||||||
|
drawUIBox(
|
||||||
|
0, FRAME_HEIGHT - DRAW_UI_TEXTBOX_HEIGHT,
|
||||||
|
DRAW_UI_TEXTBOX_WIDTH, DRAW_UI_TEXTBOX_HEIGHT,
|
||||||
|
COLOR_MAGENTA, true
|
||||||
|
);
|
||||||
|
|
||||||
|
// Title
|
||||||
|
drawText(
|
||||||
|
TEXTBOX.title,
|
||||||
|
-1,
|
||||||
|
2, FRAME_HEIGHT - DRAW_UI_TEXTBOX_HEIGHT,
|
||||||
|
COLOR_WHITE
|
||||||
|
);
|
||||||
|
|
||||||
|
// Text
|
||||||
|
drawText(
|
||||||
|
TEXTBOX.text,
|
||||||
|
TEXTBOX.textIndex,
|
||||||
|
1, FRAME_HEIGHT - DRAW_UI_TEXTBOX_HEIGHT + 1,
|
||||||
|
COLOR_WHITE
|
||||||
|
);
|
||||||
|
|
||||||
|
// Blinking cursor
|
||||||
|
memset(&FRAME_BUFFER[DRAW_UI_TEXTBOX_CURSOR_POS], ' ', 3);
|
||||||
|
memset(&FRAME_COLOR[DRAW_UI_TEXTBOX_CURSOR_POS], COLOR_WHITE, 3);
|
||||||
|
if(TEXTBOX.textIndex < TEXTBOX.textLength) return;
|
||||||
|
|
||||||
|
int32_t blink = (int32_t)(TIME.time * TEXTBOX_BLINKS_PER_SECOND) % 2;
|
||||||
|
FRAME_BUFFER[DRAW_UI_TEXTBOX_CURSOR_POS + 1] = blink ? '>' : ' ';
|
||||||
|
FRAME_BUFFER[DRAW_UI_TEXTBOX_CURSOR_POS + 2] = blink ? ' ' : '>';
|
||||||
|
}
|
@ -8,6 +8,10 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include "display/frame.h"
|
#include "display/frame.h"
|
||||||
|
|
||||||
|
#define DRAW_UI_TEXTBOX_WIDTH FRAME_WIDTH
|
||||||
|
#define DRAW_UI_TEXTBOX_HEIGHT 8
|
||||||
|
#define DRAW_UI_TEXTBOX_CURSOR_POS ((FRAME_HEIGHT * FRAME_WIDTH) - 4)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Draws a UI box to the frame buffer.
|
* Draws a UI box to the frame buffer.
|
||||||
*
|
*
|
||||||
@ -25,4 +29,9 @@ void drawUIBox(
|
|||||||
const uint16_t height,
|
const uint16_t height,
|
||||||
const uint8_t color,
|
const uint8_t color,
|
||||||
const bool_t fill
|
const bool_t fill
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Draws the UI textbox to the frame buffer.
|
||||||
|
*/
|
||||||
|
void drawUITextbox();
|
@ -6,15 +6,11 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "frame.h"
|
#include "frame.h"
|
||||||
#include "display/symbol.h"
|
|
||||||
#include "rpg/world/map.h"
|
|
||||||
#include "game/game.h"
|
#include "game/game.h"
|
||||||
#include "ui/textbox.h"
|
|
||||||
|
|
||||||
#include "display/draw/drawshape.h"
|
#include "display/draw/drawshape.h"
|
||||||
#include "display/draw/drawui.h"
|
|
||||||
#include "display/draw/drawtext.h"
|
#include "display/draw/drawtext.h"
|
||||||
#include "display/draw/drawmap.h"
|
#include "display/draw/drawstateoverworld.h"
|
||||||
|
|
||||||
char_t FRAME_BUFFER[FRAME_HEIGHT * FRAME_WIDTH];
|
char_t FRAME_BUFFER[FRAME_HEIGHT * FRAME_WIDTH];
|
||||||
uint8_t FRAME_COLOR[FRAME_HEIGHT * FRAME_WIDTH];
|
uint8_t FRAME_COLOR[FRAME_HEIGHT * FRAME_WIDTH];
|
||||||
@ -26,27 +22,22 @@ void frameInit() {
|
|||||||
void frameUpdate() {
|
void frameUpdate() {
|
||||||
switch(GAME.state) {
|
switch(GAME.state) {
|
||||||
case GAME_STATE_PAUSED:
|
case GAME_STATE_PAUSED:
|
||||||
|
const char_t *str = "PAUSED";
|
||||||
|
size_t len = strlen(str);
|
||||||
|
drawText(
|
||||||
|
str,
|
||||||
|
-1,
|
||||||
|
(FRAME_WIDTH - len) / 2,
|
||||||
|
(FRAME_HEIGHT - 1) / 2,
|
||||||
|
COLOR_WHITE
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case GAME_STATE_OVERWORLD:
|
||||||
|
drawStateOverworld();
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
map_t *map = GAME.currentMap;
|
|
||||||
if(map == NULL) return;
|
|
||||||
|
|
||||||
// Try get player
|
|
||||||
entity_t *player = mapEntityGetByType(map, ENTITY_TYPE_PLAYER);
|
|
||||||
uint16_t cameraPositionX, cameraPositionY;
|
|
||||||
if(player == NULL) {
|
|
||||||
cameraPositionX = 0;
|
|
||||||
cameraPositionY = 0;
|
|
||||||
} else {
|
|
||||||
cameraPositionX = player->x;
|
|
||||||
cameraPositionY = player->y;
|
|
||||||
}
|
|
||||||
|
|
||||||
drawMap(
|
|
||||||
GAME.currentMap,
|
|
||||||
cameraPositionX, cameraPositionY,
|
|
||||||
0, 0,
|
|
||||||
FRAME_WIDTH, FRAME_HEIGHT
|
|
||||||
);
|
|
||||||
}
|
}
|
@ -14,8 +14,6 @@
|
|||||||
#define FRAME_WIDTH FRAME_PIXEL_WIDTH/FRAME_CHAR_SIZE
|
#define FRAME_WIDTH FRAME_PIXEL_WIDTH/FRAME_CHAR_SIZE
|
||||||
#define FRAME_HEIGHT FRAME_PIXEL_HEIGHT/FRAME_CHAR_SIZE
|
#define FRAME_HEIGHT FRAME_PIXEL_HEIGHT/FRAME_CHAR_SIZE
|
||||||
|
|
||||||
#define FRAME_BOTTOM_UI_HEIGHT 8
|
|
||||||
|
|
||||||
extern char_t FRAME_BUFFER[FRAME_HEIGHT * FRAME_WIDTH];
|
extern char_t FRAME_BUFFER[FRAME_HEIGHT * FRAME_WIDTH];
|
||||||
extern uint8_t FRAME_COLOR[FRAME_HEIGHT * FRAME_WIDTH];
|
extern uint8_t FRAME_COLOR[FRAME_HEIGHT * FRAME_WIDTH];
|
||||||
|
|
||||||
|
@ -20,7 +20,7 @@ void mapInit(
|
|||||||
"Map width must be greater than 0."
|
"Map width must be greater than 0."
|
||||||
);
|
);
|
||||||
assertTrue(
|
assertTrue(
|
||||||
width < MAP_WIDTH_MAX,
|
width <= MAP_WIDTH_MAX,
|
||||||
"Map width must be less than or equal to MAP_WIDTH_MAX."
|
"Map width must be less than or equal to MAP_WIDTH_MAX."
|
||||||
);
|
);
|
||||||
assertTrue(
|
assertTrue(
|
||||||
@ -28,7 +28,7 @@ void mapInit(
|
|||||||
"Map height must be greater than 0."
|
"Map height must be greater than 0."
|
||||||
);
|
);
|
||||||
assertTrue(
|
assertTrue(
|
||||||
height < MAP_HEIGHT_MAX,
|
height <= MAP_HEIGHT_MAX,
|
||||||
"Map height must be less than or equal to MAP_HEIGHT_MAX."
|
"Map height must be less than or equal to MAP_HEIGHT_MAX."
|
||||||
);
|
);
|
||||||
assertTrue(
|
assertTrue(
|
||||||
|
@ -9,8 +9,8 @@
|
|||||||
#include "rpg/world/map.h"
|
#include "rpg/world/map.h"
|
||||||
#include "assert/assert.h"
|
#include "assert/assert.h"
|
||||||
|
|
||||||
#define TEST_MAP_WIDTH 16
|
#define TEST_MAP_WIDTH MAP_WIDTH_MAX
|
||||||
#define TEST_MAP_HEIGHT 16
|
#define TEST_MAP_HEIGHT MAP_HEIGHT_MAX
|
||||||
#define TEST_MAP_LAYERS 1
|
#define TEST_MAP_LAYERS 1
|
||||||
|
|
||||||
void testMapInit(map_t *map) {
|
void testMapInit(map_t *map) {
|
||||||
|
@ -20,8 +20,6 @@ void textboxSetText(
|
|||||||
const char_t *title,
|
const char_t *title,
|
||||||
const char_t *text
|
const char_t *text
|
||||||
) {
|
) {
|
||||||
return;
|
|
||||||
|
|
||||||
assertNotNull(text, "Text cannot be NULL.");
|
assertNotNull(text, "Text cannot be NULL.");
|
||||||
|
|
||||||
// Setup text copies
|
// Setup text copies
|
||||||
@ -32,26 +30,12 @@ void textboxSetText(
|
|||||||
|
|
||||||
// Setup title
|
// Setup title
|
||||||
if(title) {
|
if(title) {
|
||||||
len = strlen(title);
|
assertTrue(strlen(title) < TEXTBOX_TITLE_MAX, "Title is too long.");
|
||||||
assertTrue(len < TEXTBOX_TITLE_MAX, "Title is too long.");
|
|
||||||
strcpy(TEXTBOX.title, title);
|
strcpy(TEXTBOX.title, title);
|
||||||
} else {
|
} else {
|
||||||
TEXTBOX.title[0] = '\0';
|
TEXTBOX.title[0] = '\0';
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add > indicator
|
|
||||||
FRAME_BUFFER[(FRAME_HEIGHT * FRAME_WIDTH) - 4] = ' ';
|
|
||||||
FRAME_BUFFER[(FRAME_HEIGHT * FRAME_WIDTH) - 3] = '>';
|
|
||||||
FRAME_BUFFER[(FRAME_HEIGHT * FRAME_WIDTH) - 2] = ' ';
|
|
||||||
|
|
||||||
// Add title
|
|
||||||
len = strlen(TEXTBOX.title);
|
|
||||||
for(size_t i = 0; i < len; i++) {
|
|
||||||
FRAME_BUFFER[
|
|
||||||
(FRAME_HEIGHT - FRAME_BOTTOM_UI_HEIGHT) * (FRAME_WIDTH) + 2 + i
|
|
||||||
] = TEXTBOX.title[i];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Prepare for scrolling
|
// Prepare for scrolling
|
||||||
TEXTBOX.nextChar = (1.0f / TEXTBOX_CHARS_PER_SECOND);
|
TEXTBOX.nextChar = (1.0f / TEXTBOX_CHARS_PER_SECOND);
|
||||||
TEXTBOX.textIndex = 0;
|
TEXTBOX.textIndex = 0;
|
||||||
@ -63,29 +47,13 @@ bool_t textboxIsOpen() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void textboxUpdate() {
|
void textboxUpdate() {
|
||||||
return;
|
|
||||||
if(!textboxIsOpen()) return;
|
if(!textboxIsOpen()) return;
|
||||||
|
|
||||||
int32_t j;
|
|
||||||
|
|
||||||
// Have we finished scrolling?
|
// Have we finished scrolling?
|
||||||
if(TEXTBOX.textIndex >= TEXTBOX.textLength) {
|
if(TEXTBOX.textIndex >= TEXTBOX.textLength) {
|
||||||
// Wait for input
|
// Wait for input
|
||||||
if(inputWasPressed(INPUT_BIND_ACCEPT)) {
|
if(!inputWasPressed(INPUT_BIND_ACCEPT)) return;
|
||||||
TEXTBOX.open = false;
|
TEXTBOX.open = false;
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Blinking cursor
|
|
||||||
j = (int32_t)(TIME.time * TEXTBOX_BLINKS_PER_SECOND) % 2;
|
|
||||||
if(j == 0) {
|
|
||||||
FRAME_BUFFER[(FRAME_HEIGHT * FRAME_WIDTH) - 3] = '>';
|
|
||||||
FRAME_BUFFER[(FRAME_HEIGHT * FRAME_WIDTH) - 2] = ' ';
|
|
||||||
} else {
|
|
||||||
FRAME_BUFFER[(FRAME_HEIGHT * FRAME_WIDTH) - 3] = ' ';
|
|
||||||
FRAME_BUFFER[(FRAME_HEIGHT * FRAME_WIDTH) - 2] = '>';
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -93,13 +61,6 @@ void textboxUpdate() {
|
|||||||
TEXTBOX.nextChar -= TIME.delta;
|
TEXTBOX.nextChar -= TIME.delta;
|
||||||
if(TEXTBOX.nextChar > 0.0f) return;
|
if(TEXTBOX.nextChar > 0.0f) return;
|
||||||
|
|
||||||
// Scroll text
|
|
||||||
uint16_t x = TEXTBOX.textIndex % (FRAME_WIDTH - 2);
|
|
||||||
uint16_t y = TEXTBOX.textIndex / (FRAME_WIDTH - 2);
|
|
||||||
FRAME_BUFFER[
|
|
||||||
(FRAME_HEIGHT - FRAME_BOTTOM_UI_HEIGHT + 1 + y) *
|
|
||||||
(FRAME_WIDTH) + 1 + x
|
|
||||||
] = TEXTBOX.text[TEXTBOX.textIndex];
|
|
||||||
TEXTBOX.textIndex++;
|
TEXTBOX.textIndex++;
|
||||||
TEXTBOX.nextChar = (1.0f / TEXTBOX_CHARS_PER_SECOND);
|
TEXTBOX.nextChar = (1.0f / TEXTBOX_CHARS_PER_SECOND);
|
||||||
}
|
}
|
@ -7,10 +7,12 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "dawn.h"
|
#include "dawn.h"
|
||||||
#include "display/frame.h"
|
#include "display/draw/drawui.h"
|
||||||
|
|
||||||
#define TEXTBOX_TITLE_MAX 16
|
#define TEXTBOX_TITLE_MAX 16
|
||||||
#define TEXTBOX_TEXT_MAX ((FRAME_BOTTOM_UI_HEIGHT-2) * (FRAME_WIDTH - 2))
|
#define TEXTBOX_TEXT_MAX (\
|
||||||
|
(DRAW_UI_TEXTBOX_HEIGHT - 2) * (DRAW_UI_TEXTBOX_WIDTH - 2) \
|
||||||
|
)
|
||||||
#define TEXTBOX_CHARS_PER_SECOND 25
|
#define TEXTBOX_CHARS_PER_SECOND 25
|
||||||
#define TEXTBOX_BLINKS_PER_SECOND 2
|
#define TEXTBOX_BLINKS_PER_SECOND 2
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user