2023-07-09 22:02:59 +02:00
|
|
|
#include "dialogs.h"
|
|
|
|
|
|
|
|
#include <gint/keyboard.h>
|
|
|
|
#include <gint/cpu.h>
|
|
|
|
|
2023-07-13 22:23:31 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
|
2023-07-19 20:53:14 +02:00
|
|
|
#define BOX_HEIGHT (F_HEIGHT/PXSIZE+8)
|
2023-07-09 22:02:59 +02:00
|
|
|
|
2023-07-10 20:55:27 +02:00
|
|
|
extern font_t fontRPG;
|
2023-07-11 11:23:27 +02:00
|
|
|
#define FONT_USED fontRPG
|
2023-07-10 20:55:27 +02:00
|
|
|
|
2023-07-18 19:47:47 +02:00
|
|
|
#if GRAYMODEOK
|
2023-07-13 22:23:31 +02:00
|
|
|
#include <gint/gray.h>
|
|
|
|
uint32_t *lightVRAMnext, *darkVRAMnext;
|
|
|
|
uint32_t *lightVRAMcurrent, *darkVRAMcurrent;
|
|
|
|
#endif //GRAYMODEOK
|
|
|
|
|
|
|
|
|
|
|
|
void blit()
|
|
|
|
{
|
|
|
|
dupdate();
|
|
|
|
|
2023-07-18 19:47:47 +02:00
|
|
|
#if GRAYMODEOK
|
2023-07-13 22:23:31 +02:00
|
|
|
dgray_getvram( &lightVRAMnext, &darkVRAMnext );
|
|
|
|
dgray_getscreen( &lightVRAMcurrent, &darkVRAMcurrent );
|
|
|
|
|
|
|
|
memcpy( lightVRAMnext, lightVRAMcurrent, 256*sizeof( uint32_t) );
|
|
|
|
memcpy( darkVRAMnext, darkVRAMcurrent, 256*sizeof( uint32_t) );
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2023-07-10 20:55:27 +02:00
|
|
|
|
2023-07-13 12:06:48 +02:00
|
|
|
int showtext_opt(Game *game, bopti_image_t *face, char *text,
|
2023-07-19 17:43:02 +02:00
|
|
|
int call_before_end(Game *game, unsigned int i),
|
|
|
|
bool start_anim,
|
|
|
|
bool end_anim,
|
|
|
|
void for_each_screen(Game *game, unsigned int i),
|
2023-07-19 23:11:00 +02:00
|
|
|
int line_duration, bool update_screen, unsigned int start_i,
|
2023-07-19 20:06:28 +02:00
|
|
|
bool wait_continue) {
|
2023-07-11 11:23:27 +02:00
|
|
|
dfont(&FONT_USED);
|
2023-07-12 12:42:28 +02:00
|
|
|
unsigned int i, n, y = PXSIZE, l = 0;
|
2023-07-13 12:06:48 +02:00
|
|
|
int line_max_chars, return_int = 0;
|
2023-07-12 12:42:28 +02:00
|
|
|
unsigned int max_lines_amount = (BOX_HEIGHT-2)*PXSIZE/
|
|
|
|
(FONT_USED.line_height+PXSIZE);
|
2023-07-11 13:29:48 +02:00
|
|
|
const char *c;
|
2023-07-15 19:26:32 +02:00
|
|
|
if(start_anim){
|
|
|
|
/* Run a little fancy animation. */
|
2023-07-19 20:53:14 +02:00
|
|
|
for(i=0;i<=BOX_HEIGHT;i++){
|
2023-07-19 23:11:00 +02:00
|
|
|
/* Redrawing the entire screen, because maybe there was no dialog
|
|
|
|
displayed before. */
|
2023-07-19 20:06:28 +02:00
|
|
|
draw(game);
|
|
|
|
|
2023-07-19 23:11:00 +02:00
|
|
|
/* Fill the dialog box with white */
|
2023-07-15 19:26:32 +02:00
|
|
|
drect(0, 0, DWIDTH, i*PXSIZE, C_WHITE);
|
2023-07-19 23:11:00 +02:00
|
|
|
/* Draw a thick black line on the bottom of the dialog. */
|
2023-07-15 19:26:32 +02:00
|
|
|
drect(0, i*PXSIZE, DWIDTH, (i+1)*PXSIZE, C_BLACK);
|
2023-07-19 23:11:00 +02:00
|
|
|
|
|
|
|
/* Draw the part of the face of the player that can fit correctly in
|
|
|
|
* the dialog drawn. */
|
2023-07-15 19:26:32 +02:00
|
|
|
dsubimage(4*PXSIZE, 2*PXSIZE, face, 0, 0, F_WIDTH, (i-8)*PXSIZE,
|
|
|
|
DIMAGE_NONE);
|
2023-07-13 22:23:31 +02:00
|
|
|
|
2023-07-19 20:06:28 +02:00
|
|
|
dupdate();
|
|
|
|
|
2023-07-15 19:26:32 +02:00
|
|
|
while(game->frame_duration < 20) sleep();
|
|
|
|
game->frame_duration = 0;
|
|
|
|
}
|
2023-07-18 19:47:47 +02:00
|
|
|
}else{
|
2023-07-19 23:11:00 +02:00
|
|
|
/* Here I'm drawing the same as if start_anim is true, but whitout
|
|
|
|
* making an animation. */
|
2023-07-19 20:06:28 +02:00
|
|
|
draw(game);
|
2023-07-19 20:53:14 +02:00
|
|
|
drect(0, 0, DWIDTH, BOX_HEIGHT*PXSIZE, C_WHITE);
|
|
|
|
drect(0, BOX_HEIGHT*PXSIZE, DWIDTH, (BOX_HEIGHT+1)*PXSIZE, C_BLACK);
|
|
|
|
dimage(4*PXSIZE, 2*PXSIZE, face);
|
2023-07-18 19:47:47 +02:00
|
|
|
|
2023-07-19 23:11:00 +02:00
|
|
|
if(update_screen){
|
|
|
|
blit();
|
2023-07-18 21:08:12 +02:00
|
|
|
|
2023-07-19 23:11:00 +02:00
|
|
|
while(game->frame_duration < 20) sleep();
|
|
|
|
game->frame_duration = 0;
|
|
|
|
}
|
2023-07-09 22:02:59 +02:00
|
|
|
}
|
2023-07-18 19:47:47 +02:00
|
|
|
/* We should start to drawing the text on the x axis at BOX_HEIGHT to avoid
|
2023-07-09 22:02:59 +02:00
|
|
|
* drawing on the face. */
|
2023-07-19 17:43:02 +02:00
|
|
|
for(i=start_i;i<strlen(text);i++){
|
2023-07-19 20:06:28 +02:00
|
|
|
if(!l && for_each_screen) for_each_screen(game, i);
|
2023-07-11 11:23:27 +02:00
|
|
|
/* Get how many chars we can draw on screen with a padding on the left
|
2023-07-11 13:29:48 +02:00
|
|
|
* of BOX_HEIGHT px and on the right of 1 px. */
|
|
|
|
c = drsize(text+i, &FONT_USED, DWIDTH-(BOX_HEIGHT*PXSIZE+PXSIZE), NULL);
|
|
|
|
/* c is a pointer to the last char that can be drawn. So: */
|
|
|
|
line_max_chars = c-(text+i);
|
2023-07-19 20:06:28 +02:00
|
|
|
/* TODO: Handle lines that are longer than what I can draw and '\n'. */
|
2023-07-11 11:23:27 +02:00
|
|
|
/* Loop from the end to the start for word wrap. */
|
2023-07-19 20:06:28 +02:00
|
|
|
if(*c){
|
|
|
|
/* If we are not drawing the end of the text. */
|
2023-07-18 19:47:47 +02:00
|
|
|
for(n=line_max_chars; n>0; n--) {
|
2023-07-19 17:43:02 +02:00
|
|
|
/* If we found a space, we can draw this line and do the same
|
|
|
|
* for the next line. */
|
2023-07-18 19:47:47 +02:00
|
|
|
if(text[i+n] == ' '){
|
|
|
|
dtext_opt(BOX_HEIGHT*PXSIZE, y, C_BLACK, C_NONE, DTEXT_LEFT,
|
|
|
|
DTEXT_TOP, text+i, n); /* Draw everything. */
|
|
|
|
/* Increment y by the line height. */
|
|
|
|
y += FONT_USED.line_height+PXSIZE;
|
|
|
|
i += n; /* We drew everything to i+n */
|
|
|
|
l++; /* We drew one more line. */
|
|
|
|
break;
|
|
|
|
}
|
2023-07-11 11:23:27 +02:00
|
|
|
}
|
2023-07-18 19:47:47 +02:00
|
|
|
}else{
|
2023-07-19 20:06:28 +02:00
|
|
|
/* If it is the last line of the text. */
|
2023-07-18 19:47:47 +02:00
|
|
|
dtext_opt(BOX_HEIGHT*PXSIZE, y, C_BLACK, C_NONE, DTEXT_LEFT,
|
|
|
|
DTEXT_TOP, text+i, line_max_chars);
|
|
|
|
y += FONT_USED.line_height+PXSIZE;
|
|
|
|
i += line_max_chars;
|
|
|
|
l++;
|
2023-07-11 11:23:27 +02:00
|
|
|
}
|
|
|
|
if(l>=max_lines_amount-1){
|
|
|
|
/* We drew one entire screen, reset everything to draw the next one.
|
|
|
|
*/
|
|
|
|
/* Make a little animation :). */
|
2023-07-19 23:11:00 +02:00
|
|
|
if(update_screen) blit();
|
2023-07-19 17:43:02 +02:00
|
|
|
while(game->frame_duration < line_duration) sleep();
|
2023-07-11 11:23:27 +02:00
|
|
|
game->frame_duration = 0;
|
|
|
|
/* Ask the user to press EXE to continue. */
|
2023-07-12 12:42:28 +02:00
|
|
|
dtext(BOX_HEIGHT*PXSIZE, y, C_BLACK, "[EXE] to continue ...");
|
2023-07-11 11:23:27 +02:00
|
|
|
}
|
|
|
|
/* Make a little animation :). */
|
2023-07-19 23:11:00 +02:00
|
|
|
if(update_screen) blit();
|
2023-07-11 11:23:27 +02:00
|
|
|
if(l>=max_lines_amount-1){
|
2023-07-19 23:11:00 +02:00
|
|
|
/* If we drew one entire screen. */
|
|
|
|
/* Wait that the EXE key is pressed if we should. */
|
2023-07-19 20:06:28 +02:00
|
|
|
if(wait_continue) while(getkey().key != KEY_EXE) sleep();
|
|
|
|
/* Clear the text area. */
|
|
|
|
drect(BOX_HEIGHT*PXSIZE, 0, DWIDTH, (BOX_HEIGHT-1)*PXSIZE-2,
|
2023-07-12 12:42:28 +02:00
|
|
|
C_WHITE);
|
2023-07-11 13:29:48 +02:00
|
|
|
/* Reset y and l. */
|
2023-07-12 12:42:28 +02:00
|
|
|
y = PXSIZE;
|
2023-07-11 13:29:48 +02:00
|
|
|
l = 0;
|
2023-07-11 11:23:27 +02:00
|
|
|
}
|
|
|
|
else{
|
2023-07-19 23:11:00 +02:00
|
|
|
/* Else, wait a bit for the animation. */
|
2023-07-19 17:43:02 +02:00
|
|
|
while(game->frame_duration < line_duration) sleep();
|
2023-07-12 12:42:28 +02:00
|
|
|
game->frame_duration = 0;
|
2023-07-11 11:23:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if(l<max_lines_amount-1){
|
|
|
|
/* If we have not filled everthing with text at the end. */
|
|
|
|
/* Make a little animation :). */
|
2023-07-19 23:11:00 +02:00
|
|
|
if(update_screen) blit();
|
2023-07-19 17:43:02 +02:00
|
|
|
while(game->frame_duration < line_duration) sleep();
|
2023-07-11 11:23:27 +02:00
|
|
|
game->frame_duration = 0;
|
|
|
|
/* Ask the user to press EXE to continue. */
|
2023-07-14 15:28:00 +02:00
|
|
|
dtext(BOX_HEIGHT*PXSIZE, y, C_BLACK, "[EXE] to continue ...");
|
2023-07-19 23:11:00 +02:00
|
|
|
/* Update the screen and wait for EXE being pressed, if needed. */
|
|
|
|
if(update_screen) blit();
|
2023-07-19 20:06:28 +02:00
|
|
|
if(wait_continue) while(getkey().key != KEY_EXE) sleep();
|
2023-07-09 22:02:59 +02:00
|
|
|
}
|
2023-07-19 17:43:02 +02:00
|
|
|
if(call_before_end) return_int = call_before_end(game, i);
|
2023-07-15 19:26:32 +02:00
|
|
|
if(end_anim){
|
|
|
|
/* Run another little fancy animation. */
|
2023-07-19 20:53:14 +02:00
|
|
|
for(i=BOX_HEIGHT;i>0;i--){
|
2023-07-15 19:26:32 +02:00
|
|
|
draw(game);
|
|
|
|
drect(0, 0, DWIDTH, i*PXSIZE, C_WHITE);
|
|
|
|
drect(0, i*PXSIZE, DWIDTH, (i+1)*PXSIZE, C_BLACK);
|
|
|
|
dsubimage(4*PXSIZE, 2*PXSIZE, face, 0, 0, F_WIDTH, (i-8)*PXSIZE,
|
|
|
|
DIMAGE_NONE);
|
|
|
|
|
|
|
|
dupdate();
|
|
|
|
|
|
|
|
while(game->frame_duration < 20) sleep();
|
|
|
|
game->frame_duration = 0;
|
|
|
|
}
|
2023-07-09 22:02:59 +02:00
|
|
|
}
|
2023-07-13 12:06:48 +02:00
|
|
|
return return_int;
|
|
|
|
}
|
|
|
|
|
2023-07-19 17:43:02 +02:00
|
|
|
void showtext_dialog(Game *game, bopti_image_t *face, char *text,
|
|
|
|
bool dialog_start, bool dialog_end) {
|
|
|
|
showtext_opt(game, face, text, NULL, dialog_start, dialog_end, NULL, 100,
|
2023-07-19 20:06:28 +02:00
|
|
|
true, 0, true);
|
2023-07-13 12:06:48 +02:00
|
|
|
}
|
|
|
|
|
2023-07-15 11:43:24 +02:00
|
|
|
#define CHOICE_BOX_HEIGHT 10
|
|
|
|
#define CHOICE_BOX_PADDING_TOP 3
|
2023-07-13 12:06:48 +02:00
|
|
|
|
2023-07-19 17:43:02 +02:00
|
|
|
char *_choices, *_text;
|
2023-07-15 19:26:32 +02:00
|
|
|
int _choices_amount, _default_choice;
|
2023-07-19 17:43:02 +02:00
|
|
|
bopti_image_t *_face;
|
|
|
|
unsigned int _i;
|
2023-07-15 11:43:24 +02:00
|
|
|
|
2023-07-19 17:43:02 +02:00
|
|
|
void _choice_screen_call(Game *game, unsigned int i) {
|
|
|
|
_i = i;
|
2023-07-18 19:47:47 +02:00
|
|
|
}
|
|
|
|
|
2023-07-19 17:43:02 +02:00
|
|
|
int _choice_call_before_end(Game *game, unsigned int org_i) {
|
2023-07-15 11:43:24 +02:00
|
|
|
int i, key;
|
|
|
|
/* Make a little animation because we looove little animations ;) */
|
|
|
|
for(i=0;i<DWIDTH/8+1;i++){
|
2023-07-19 20:53:14 +02:00
|
|
|
drect(0, (BOX_HEIGHT+1)*PXSIZE+1, i*(DWIDTH/8),
|
2023-07-15 11:43:24 +02:00
|
|
|
(BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, C_WHITE);
|
2023-07-19 16:57:40 +02:00
|
|
|
drect(i*(DWIDTH/8), BOX_HEIGHT*PXSIZE, i*(DWIDTH/8)+PXSIZE-1,
|
|
|
|
(BOX_HEIGHT+CHOICE_BOX_HEIGHT+1)*PXSIZE, C_BLACK);
|
2023-07-19 20:53:14 +02:00
|
|
|
drect(0, (BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, i*(DWIDTH/8),
|
2023-07-15 11:43:24 +02:00
|
|
|
(BOX_HEIGHT+CHOICE_BOX_HEIGHT+1)*PXSIZE, C_BLACK);
|
|
|
|
blit();
|
|
|
|
while(game->frame_duration < 20) sleep();
|
|
|
|
game->frame_duration = 0;
|
|
|
|
}
|
2023-07-15 19:26:32 +02:00
|
|
|
const int choice_size = DWIDTH/_choices_amount;
|
2023-07-18 21:08:12 +02:00
|
|
|
int arrow_width, arrow_height, selected = _default_choice, pos = 0;
|
|
|
|
dsize(">", &FONT_USED, &arrow_width, &arrow_height);
|
|
|
|
arrow_width += FONT_USED.char_spacing;
|
2023-07-15 11:43:24 +02:00
|
|
|
do{
|
2023-07-18 19:47:47 +02:00
|
|
|
/* Display the diffrent choices. */
|
2023-07-15 19:26:32 +02:00
|
|
|
for(i=0;i<_choices_amount;i++){
|
|
|
|
if(i == selected) dtext(i*choice_size+PXSIZE,
|
|
|
|
(BOX_HEIGHT+CHOICE_BOX_PADDING_TOP)*PXSIZE,
|
2023-07-15 11:43:24 +02:00
|
|
|
C_BLACK, ">");
|
2023-07-15 19:26:32 +02:00
|
|
|
dtext(i*choice_size+arrow_width+PXSIZE,
|
|
|
|
(BOX_HEIGHT+CHOICE_BOX_PADDING_TOP)*PXSIZE, C_BLACK,
|
2023-07-18 19:47:47 +02:00
|
|
|
_choices+pos+i);
|
|
|
|
pos += strlen(_choices+pos);
|
2023-07-15 11:43:24 +02:00
|
|
|
}
|
|
|
|
blit();
|
|
|
|
key = getkey().key;
|
2023-07-18 21:08:12 +02:00
|
|
|
if(key == KEY_LEFT && selected > 0){
|
|
|
|
/* Remove the old arrow. */
|
|
|
|
drect(selected*choice_size+PXSIZE,
|
|
|
|
(BOX_HEIGHT+CHOICE_BOX_PADDING_TOP)*PXSIZE,
|
|
|
|
selected*choice_size+PXSIZE+arrow_width,
|
|
|
|
(BOX_HEIGHT+CHOICE_BOX_PADDING_TOP)*PXSIZE+arrow_height,
|
|
|
|
C_WHITE);
|
|
|
|
selected--;
|
|
|
|
}
|
2023-07-19 16:57:40 +02:00
|
|
|
else if(key == KEY_RIGHT && selected < _choices_amount-1){
|
2023-07-18 21:08:12 +02:00
|
|
|
/* Remove the old arrow. */
|
|
|
|
drect(selected*choice_size+PXSIZE,
|
|
|
|
(BOX_HEIGHT+CHOICE_BOX_PADDING_TOP)*PXSIZE,
|
|
|
|
selected*choice_size+PXSIZE+arrow_width,
|
|
|
|
(BOX_HEIGHT+CHOICE_BOX_PADDING_TOP)*PXSIZE+arrow_height,
|
|
|
|
C_WHITE);
|
|
|
|
selected++;
|
|
|
|
}
|
2023-07-15 11:43:24 +02:00
|
|
|
}while(key != KEY_EXE);
|
|
|
|
/* Make a little animation because we looove little animations ;) */
|
2023-07-15 19:26:32 +02:00
|
|
|
for(i=DWIDTH/8+1;i>0;i--){
|
2023-07-19 17:43:02 +02:00
|
|
|
draw(game);
|
|
|
|
showtext_opt(game, _face, _text, NULL, false, false, NULL, 0, false,
|
2023-07-19 20:06:28 +02:00
|
|
|
_i, false);
|
2023-07-19 20:53:14 +02:00
|
|
|
drect(0, (BOX_HEIGHT+1)*PXSIZE+1, i*(DWIDTH/8),
|
2023-07-15 11:43:24 +02:00
|
|
|
(BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, C_WHITE);
|
2023-07-19 16:57:40 +02:00
|
|
|
drect(i*(DWIDTH/8), BOX_HEIGHT*PXSIZE, i*(DWIDTH/8)+PXSIZE-1,
|
|
|
|
(BOX_HEIGHT+CHOICE_BOX_HEIGHT+1)*PXSIZE, C_BLACK);
|
2023-07-19 20:53:14 +02:00
|
|
|
drect(0, (BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, i*(DWIDTH/8),
|
2023-07-15 11:43:24 +02:00
|
|
|
(BOX_HEIGHT+CHOICE_BOX_HEIGHT+1)*PXSIZE, C_BLACK);
|
2023-07-19 17:43:02 +02:00
|
|
|
dupdate();
|
2023-07-15 11:43:24 +02:00
|
|
|
while(game->frame_duration < 20) sleep();
|
|
|
|
game->frame_duration = 0;
|
|
|
|
}
|
|
|
|
return selected;
|
2023-07-09 22:02:59 +02:00
|
|
|
}
|
2023-07-15 19:26:32 +02:00
|
|
|
|
|
|
|
int showtext_dialog_ask(Game *game, bopti_image_t *face, char *text, bool start,
|
|
|
|
bool end, char *choices, int choices_amount,
|
|
|
|
int default_choice) {
|
|
|
|
_choices = choices;
|
|
|
|
_choices_amount = choices_amount;
|
|
|
|
_default_choice = default_choice;
|
2023-07-19 17:43:02 +02:00
|
|
|
_face = face;
|
|
|
|
_text = text;
|
|
|
|
return showtext_opt(game, face, text, _choice_call_before_end, start, end,
|
2023-07-19 20:06:28 +02:00
|
|
|
_choice_screen_call, 100, true, 0, true);
|
2023-07-15 19:26:32 +02:00
|
|
|
}
|