Collab_RPG/src/dialogs.c

269 lines
9.3 KiB
C
Raw Normal View History

#include "dialogs.h"
#include <gint/keyboard.h>
#include <gint/cpu.h>
#include <gint/image.h>
#include <string.h>
#include "config.h"
#define BOX_HEIGHT (F_HEIGHT/PXSIZE+8)
extern font_t fontRPG;
#define FONT_USED fontRPG
#if GRAYMODEOK
#include <gint/gray.h>
uint32_t *lightVRAMnext, *darkVRAMnext;
uint32_t *lightVRAMcurrent, *darkVRAMcurrent;
#endif //GRAYMODEOK
void blit()
{
dupdate();
#if GRAYMODEOK
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-13 12:06:48 +02:00
int showtext_opt(Game *game, bopti_image_t *face, char *text,
int call_before_end(Game *game), bool start_anim,
bool end_anim) {
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;
draw(game);
if(start_anim){
/* Run a little fancy animation. */
for(i=0;i<BOX_HEIGHT;i++){
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;
}
}else{
drect(0, 0, DWIDTH, (BOX_HEIGHT-1)*PXSIZE, C_WHITE);
drect(0, (BOX_HEIGHT-1)*PXSIZE, DWIDTH, BOX_HEIGHT*PXSIZE, C_BLACK);
dsubimage(4*PXSIZE, 2*PXSIZE, face, 0, 0, F_WIDTH,
(BOX_HEIGHT-7)*PXSIZE, DIMAGE_NONE);
dupdate();
}
/* We should start to drawing the text on the x axis at BOX_HEIGHT to avoid
* drawing on the face. */
for(i=0;i<strlen(text);i++){
/* 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);
/* TODO: Handle lines that are longer than what I can draw. */
/* Loop from the end to the start for word wrap. */
if(*(c+1) != '\0'){
for(n=line_max_chars; n>0; n--) {
/* If we found a space, we can draw this line and do the same for
* the next line. */
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;
}
}
}else{
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++;
}
if(l>=max_lines_amount-1){
/* We drew one entire screen, reset everything to draw the next one.
*/
/* Make a little animation :). */
blit();
while(game->frame_duration < 100) sleep();
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 ...");
}
/* Make a little animation :). */
blit();
if(l>=max_lines_amount-1){
while(getkey().key != KEY_EXE) sleep();
/* Clear the screen. */
2023-07-12 12:42:28 +02:00
drect(BOX_HEIGHT*PXSIZE, 0, DWIDTH, (BOX_HEIGHT-1)*PXSIZE-1,
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;
}
else{
while(game->frame_duration < 100) sleep();
2023-07-12 12:42:28 +02:00
game->frame_duration = 0;
}
}
if(l<max_lines_amount-1){
/* If we have not filled everthing with text at the end. */
/* Make a little animation :). */
blit();
while(game->frame_duration < 100) sleep();
game->frame_duration = 0;
/* Ask the user to press EXE to continue. */
dtext(BOX_HEIGHT*PXSIZE, y, C_BLACK, "[EXE] to continue ...");
blit();
while(getkey().key != KEY_EXE) sleep();
}
if(call_before_end) return_int = call_before_end(game);
if(end_anim){
/* Run another little fancy animation. */
for(i=40;i>0;i--){
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-13 12:06:48 +02:00
return return_int;
}
void showtext(Game *game, bopti_image_t *face, char *text) {
showtext_opt(game, face, text, NULL, true, true);
}
void showtext_dialog_start(Game *game, bopti_image_t *face, char *text) {
showtext_opt(game, face, text, NULL, true, false);
}
void showtext_dialog_mid(Game *game, bopti_image_t *face, char *text) {
showtext_opt(game, face, text, NULL, false, false);
}
void showtext_dialog_end(Game *game, bopti_image_t *face, char *text) {
showtext_opt(game, face, text, NULL, false, true);
}
#define CHOICE_BOX_HEIGHT 10
#define CHOICE_BOX_PADDING_TOP 3
2023-07-13 12:06:48 +02:00
char *_choices;
int _choices_amount, _default_choice;
/* store_vram_part(int x1, int y1, int x2, int y2) and restore_vram_part(int x,
* int y) are needed in the animations of the interaction dialog. */
#ifdef FXCG50
image_t vram_part;
#else
// Really need to code this!
#endif
void store_vram_part(int x1, int y1, int x2, int y2) {
#ifdef FXCG50
image_sub(image_create_vram(), x1, y1, x2-x1, y2-y1, &vram_part);
#else
// Really need to code this!
#endif
}
void restore_vram_part(int x, int y) {
#ifdef FXCG50
dimage(x, y, &vram_part);
#else
// Really need to code this!
#endif
}
int _choice_call_before_end(Game *game) {
int i, key;
/* Make a little animation because we looove little animations ;) */
store_vram_part(0, (BOX_HEIGHT+1)*PXSIZE,
DWIDTH, (BOX_HEIGHT+CHOICE_BOX_HEIGHT+1)*PXSIZE);
for(i=0;i<DWIDTH/8+1;i++){
drect(0, BOX_HEIGHT*PXSIZE, i*(DWIDTH/8),
(BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, C_WHITE);
drect(i*(DWIDTH/8), BOX_HEIGHT*PXSIZE, i*(DWIDTH/8)+PXSIZE,
(BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, C_BLACK);
drect(0, (BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, i*(DWIDTH/8),
(BOX_HEIGHT+CHOICE_BOX_HEIGHT+1)*PXSIZE, C_BLACK);
blit();
while(game->frame_duration < 20) sleep();
game->frame_duration = 0;
}
const int choice_size = DWIDTH/_choices_amount;
int arrow_width, selected = _default_choice, pos = 0;
dsize(">", &FONT_USED, &arrow_width, NULL);
do{
/* Clear the box. */
drect(0, (BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, DWIDTH,
(BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE-1, C_WHITE);
/* Display the diffrent choices. */
for(i=0;i<_choices_amount;i++){
if(i == selected) dtext(i*choice_size+PXSIZE,
(BOX_HEIGHT+CHOICE_BOX_PADDING_TOP)*PXSIZE,
C_BLACK, ">");
dtext(i*choice_size+arrow_width+PXSIZE,
(BOX_HEIGHT+CHOICE_BOX_PADDING_TOP)*PXSIZE, C_BLACK,
_choices+pos+i);
pos += strlen(_choices+pos);
}
blit();
key = getkey().key;
if(key == KEY_LEFT && selected > 0) selected--;
else if(key == KEY_RIGHT && selected < _choices_amount) selected++;
}while(key != KEY_EXE);
/* Make a little animation because we looove little animations ;) */
for(i=DWIDTH/8+1;i>0;i--){
restore_vram_part(0, BOX_HEIGHT*PXSIZE);
drect(0, BOX_HEIGHT*PXSIZE, i*(DWIDTH/8),
(BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, C_WHITE);
drect(i*(DWIDTH/8), BOX_HEIGHT*PXSIZE, i*(DWIDTH/8)+PXSIZE,
(BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, C_BLACK);
drect(0, (BOX_HEIGHT+CHOICE_BOX_HEIGHT)*PXSIZE, i*(DWIDTH/8),
(BOX_HEIGHT+CHOICE_BOX_HEIGHT+1)*PXSIZE, C_BLACK);
blit();
while(game->frame_duration < 20) sleep();
game->frame_duration = 0;
}
#ifdef FXCG50
//
#else
// Really need to code this!
#endif
return selected;
}
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;
return showtext_opt(game, face, text, _choice_call_before_end, start, end);
}