steed/src/steed/game.c

3235 lines
64 KiB
C

#include "externals.h"
#include "internals.h"
#include "list.h"
#include "idf.h"
int game_running = 1;
char *err_msg = NULL;
#define ERR_MSG_MAX 512
char game_cwd[PATH_MAX];
char *curgame_dir = NULL;
#define DATA_IDF "data.idf"
idf_t game_idf = NULL;
int game_own_theme = 0;
int game_theme_changed = 0;
static int game_pic_w = 0;
static int game_pic_h = 0;
static char *last_music = NULL;
static char *last_pict = NULL;
static char *last_title = NULL;
static char *last_cmd = NULL;
void game_cursor(int on);
void mouse_reset(int hl);
void game_err_msg(const char *s)
{
if (err_msg)
free(err_msg);
if (s) {
err_msg = strdup(s);
if (err_msg && strlen(err_msg) > ERR_MSG_MAX) {
err_msg[ERR_MSG_MAX - 4] = 0;
strcat(err_msg, "...");
}
} else
err_msg = NULL;
}
static int is_game(const char *path, const char *n)
{
int rc = 0;
char *p = getfilepath(path, n);
char *pp;
if (!p)
return 0;
if (idf_magic(p)) {
free(p);
return 1;
}
strcat(p, "/");
pp = malloc(strlen(p) + strlen(MAIN_FILE) + 1);
if (pp) {
strcpy(pp, p);
strcat(pp, MAIN_FILE);
if (!access(pp, R_OK))
rc = 1;
free(pp);
}
free(p);
return rc;
}
struct game *games = NULL;
int games_nr = 0;
void free_last(void);
static struct game *game_lookup(const char *name)
{
int i;
if (!name || !*name) {
if (games_nr == 1)
return &games[0];
return NULL;
}
for (i = 0; i<games_nr; i ++) {
if (!strlowcmp(games[i].dir, name)) {
return &games[i];
}
}
return NULL;
}
int game_reset(void)
{
game_release_theme();
free_last();
if (game_select(curgame_dir))
goto out;
if (game_apply_theme())
goto out;
return 0;
out:
game_done(0);
if (game_init(NULL)) {
game_error("");
return -1;
}
return -1;
}
int game_select(const char *name)
{
int rc;
struct game *g;
FREE(last_cmd);
game_stop_mus(500);
g = game_lookup(name);
if ((!name || !*name) && !g) {
game_use_theme();
return game_theme_init();
}
if (setdir(game_cwd))
return -1;
if (g) {
char *oldgame = curgame_dir;
curgame_dir = g->dir;
instead_done();
if (instead_init()) {
curgame_dir = oldgame;
return -1;
}
if (g->idf) {
setdir(game_cwd);
if (oldgame != curgame_dir) {
idf_done(game_idf);
game_idf = idf_init(g->path);
if (game_idf)
idf_only(game_idf, 1);
}
}
if ((!g->idf && setdir(g->path)) || (g->idf && !game_idf)) {
curgame_dir = oldgame;
return -1;
}
if (oldgame != curgame_dir && !g->idf) {
idf_done(game_idf);
game_idf = idf_init(DATA_IDF);
}
game_use_theme();
if (game_theme_init()) {
curgame_dir = oldgame;
return -1;
}
if (instead_load(MAIN_FILE)) {
curgame_dir = oldgame;
return -1;
}
rc = instead_function("game:ini", NULL); instead_clear();
return rc;
} else {
game_use_theme();
game_theme_init();
}
return 0;
}
static char *game_name(const char *path, const char *d_name)
{
char *l;
char *p;
if (idf_magic(path)) {
idf_t idf = idf_init(path);
idff_t idff = idf_open(idf, MAIN_FILE);
l = lookup_lang_tag_idf(idff, "Name", "--");
idf_close(idff);
idf_done(idf);
if (l)
return l;
goto err;
}
p = getfilepath(path, MAIN_FILE);
if (!p)
goto err;
l = lookup_lang_tag(p, "Name", "--");
free(p);
if (l)
return l;
err:
return strdup(d_name);
}
int games_rename(void)
{
int i;
char cwd[PATH_MAX];
getdir(cwd, sizeof(cwd));
setdir(game_cwd);
for (i = 0; i < games_nr; i++) {
FREE(games[i].name);
games[i].name = game_name(dirpath(games[i].path), games[i].dir);
}
setdir(cwd);
return 0;
}
static int cmp_game(const void *p1, const void *p2)
{
const struct game *g1 = (const struct game*)p1;
const struct game *g2 = (const struct game*)p2;
int g1_s = !!strncmp(g1->path, GAMES_PATH, strlen(GAMES_PATH));
int g2_s = !!strncmp(g2->path, GAMES_PATH, strlen(GAMES_PATH));
if (g1_s != g2_s)
return g1_s - g2_s;
return strcmp(g1->name, g2->name);
}
static void games_sort()
{
qsort(games, games_nr, sizeof(struct game), cmp_game);
}
static int games_add(const char *path, const char *dir)
{
char *p;
if (!is_game(path, dir))
return -1;
p = getfilepath(path, dir);
if (!p)
return -1;
if (!idf_magic(p)) {
strcat(p, "/");
games[games_nr].idf = 0;
} else
games[games_nr].idf = 1;
games[games_nr].path = p;
games[games_nr].dir = strdup(dir);
games[games_nr].name = game_name(p, dir);
games_nr ++;
return 0;
}
int games_replace(const char *path, const char *dir)
{
int rc;
char *p;
struct game *g;
if (!is_game(path, dir))
return -1;
g = game_lookup(dir);
if (g) {
if (g->idf)
p = getfilepath(path, dir);
else
p = getpath(path, dir);
if (!p)
return -1;
free(g->path);
free(g->dir);
free(g->name);
g->path = p;
g->dir = strdup(dir);
g->name = game_name(p, dir);
games_sort();
return 0;
}
games = realloc(games, sizeof(struct game) * (1 + games_nr));
if (!games)
return -1;
rc = games_add(path, dir);
if (!rc)
games_sort();
return rc;
}
int games_lookup(const char *path)
{
int n = 0, i = 0;
DIR *d;
struct dirent *de;
if (!path)
return 0;
d = opendir(path);
if (!d)
return -1;
while ((de = readdir(d))) {
/*if (de->d_type != DT_DIR)
continue;*/
if (game_lookup(de->d_name))
continue;
if (!is_game(path, de->d_name))
continue;
n ++;
}
rewinddir(d);
if (!n)
goto out;
games = realloc(games, sizeof(struct game) * (n + games_nr));
while ((de = readdir(d)) && i < n) {
/*if (de->d_type != DT_DIR)
continue;*/
if (game_lookup(de->d_name))
continue;
if (games_add(path, de->d_name))
continue;
i ++;
}
out:
closedir(d);
games_sort();
return 0;
}
int games_remove(int gtr)
{
int rc;
rc = remove_dir(games[gtr].path);
free(games[gtr].name); free(games[gtr].dir); free(games[gtr].path);
games_nr --;
memmove(&games[gtr], &games[gtr + 1], (games_nr - gtr) * sizeof(struct game));
games = realloc(games, games_nr * sizeof(struct game));
return rc;
}
static int motion_mode = 0;
static int motion_id = 0;
static int motion_y = 0;
static int mx, my;
static img_t menubg = NULL;
static img_t menu = NULL;
static int menu_shown = 0;
static int browse_dialog = 0;
int game_cmd(char *cmd);
void game_clear(int x, int y, int w, int h)
{
game_cursor(CURSOR_CLEAR);
if (game_theme.bg)
gfx_draw_bg(game_theme.bg, x, y, w, h);
else
gfx_clear(x, y, w, h);
if (menu_shown) {
int xx = x - mx;
int yy = y - my;
gfx_draw_from(menubg, xx, yy, w, h, NULL, x, y);
gfx_draw_from(menu, xx, yy, w, h, NULL, x, y);
}
}
void game_clear(int x, int y, int w, int h);
struct el {
int id;
int x;
int y;
int mx;
int my; /* coordinates */
int type;
int drawn;
// int clone;
union {
layout_t lay;
textbox_t box;
img_t img;
void *p;
} p;
};
enum {
elt_box,
elt_layout,
elt_image,
};
enum {
el_menu = 1,
el_title,
el_ways,
el_inv,
el_scene,
el_sup,
el_sdown,
// el_sslide,
el_iup,
el_idown,
// el_islide,
el_spic,
el_menu_button,
el_max,
};
static struct el objs[el_max];
static void el_set(int i, int t, int x, int y, void *p)
{
objs[i].id = i;
objs[i].x = x;
objs[i].y = y;
objs[i].p.p = p;
objs[i].type = t;
objs[i].drawn = 0;
// objs[i].clone = 0;
}
static struct el *el(int num)
{
return &objs[num];
}
static textbox_t el_box(int num)
{
return objs[num].p.box;
}
static layout_t el_layout(int num)
{
return objs[num].p.lay;
}
static img_t el_img(int num)
{
return objs[num].p.img;
}
char *game_menu_gen(void);
void game_menu(int nr)
{
cur_menu = nr;
game_menu_box(1, game_menu_gen());
}
int game_error(const char *name)
{
game_done(1);
if (game_init(NULL)) {
fprintf(stderr,"Fatal error! Can't init anything!!!\n");
exit(1);
}
game_menu(menu_error);
return 0;
}
static void el_draw(int n);
int window_sw = 0;
int fullscreen_sw = 0;
int game_load(int nr)
{
char *s;
s = game_save_path(0, nr);
if (s && !access(s, R_OK)) {
char cmd[PATH_MAX];
char sav[PATH_MAX];
strcpy(sav, s);
snprintf(cmd, sizeof(cmd) - 1, "load %s", s);
game_cmd(cmd);
if (nr == -1)
unlink(sav);
return 0;
}
return -1;
}
int game_saves_enabled(void)
{
int rc;
instead_function("instead.isEnableSave", NULL);
rc = instead_bretval(0);
instead_clear();
return rc;
}
int game_autosave_enabled(void)
{
int rc;
instead_function("instead.isEnableAutosave", NULL);
rc = instead_bretval(0);
instead_clear();
return rc;
}
int game_save(int nr)
{
char *s = game_save_path(1, nr);
char cmd[PATH_MAX];
char *p;
if (s) {
if (nr == -1 || nr == 0) {
struct instead_args args_1[] = {
{ .val = "-1", .type = INSTEAD_NUM },
{ .val = NULL, }
};
struct instead_args args_0[] = {
{ .val = "0", .type = INSTEAD_NUM },
{ .val = NULL, }
};
if (nr == -1)
instead_function("instead.autosave", args_1); /* enable saving for -1 */
else if (!game_autosave_enabled())
return 0; /* nothing todo */
else
instead_function("instead.autosave", args_0); /* enable saving for 0 */
instead_clear();
}
snprintf(cmd, sizeof(cmd) - 1, "save %s", s);
p = instead_cmd(cmd);
if (p)
free(p);
if (!instead_bretval(1) || (!p && err_msg)) {
instead_clear();
game_menu(menu_warning);
return -1;
}
instead_clear();
return 0;
}
return -1;
}
static int inv_enabled(void)
{
return (game_theme.inv_mode != INV_MODE_DISABLED);
}
int game_apply_theme(void)
{
layout_t lay;
textbox_t box;
memset(objs, 0, sizeof(struct el) * el_max);
gfx_bg(game_theme.bgcol);
if (!DIRECT_MODE)
game_clear(0, 0, game_theme.w, game_theme.h);
gfx_flip();
lay = txt_layout(game_theme.font, game_theme.win_align, game_theme.win_w, game_theme.win_h);
if (!lay)
return -1;
box = txt_box(game_theme.win_w, game_theme.win_h);
if (!box)
return -1;
txt_layout_color(lay, game_theme.fgcol);
txt_layout_link_color(lay, game_theme.lcol);
// txt_layout_link_style(lay, 4);
txt_layout_active_color(lay, game_theme.acol);
txt_layout_font_height(lay, game_theme.font_height);
txt_box_set(box, lay);
el_set(el_scene, elt_box, game_theme.win_x, 0, box);
if (inv_enabled()) {
lay = txt_layout(game_theme.inv_font, INV_ALIGN(game_theme.inv_mode),
game_theme.inv_w, game_theme.inv_h);
if (!lay)
return -1;
txt_layout_color(lay, game_theme.icol);
txt_layout_link_color(lay, game_theme.ilcol);
txt_layout_active_color(lay, game_theme.iacol);
txt_layout_font_height(lay, game_theme.inv_font_height);
box = txt_box(game_theme.inv_w, game_theme.inv_h);
if (!box)
return -1;
txt_box_set(box, lay);
el_set(el_inv, elt_box, game_theme.inv_x, game_theme.inv_y, box);
} else
el_set(el_inv, elt_box, game_theme.inv_x, game_theme.inv_y, NULL);
lay = txt_layout(game_theme.font, ALIGN_CENTER, game_theme.win_w, 0);
if (!lay)
return -1;
txt_layout_color(lay, game_theme.fgcol);
txt_layout_link_color(lay, game_theme.lcol);
txt_layout_active_color(lay, game_theme.acol);
txt_layout_font_height(lay, game_theme.font_height);
el_set(el_title, elt_layout, game_theme.win_x, game_theme.win_y, lay);
lay = txt_layout(game_theme.font, ALIGN_CENTER, game_theme.win_w, 0);
if (!lay)
return -1;
txt_layout_color(lay, game_theme.fgcol);
txt_layout_link_color(lay, game_theme.lcol);
txt_layout_active_color(lay, game_theme.acol);
txt_layout_font_height(lay, game_theme.font_height);
el_set(el_ways, elt_layout, game_theme.win_x, 0, lay);
el_set(el_sdown, elt_image, 0, 0, game_theme.a_down);
el_set(el_sup, elt_image, 0, 0, game_theme.a_up);
el_set(el_idown, elt_image, 0, 0, game_theme.inv_a_down);
el_set(el_iup, elt_image, 0, 0, game_theme.inv_a_up);
el_set(el_spic, elt_image, game_theme.win_x, game_theme.win_y, NULL);
el_set(el_menu, elt_layout, 0, 0, NULL);
el_set(el_menu_button, elt_image, game_theme.menu_button_x, game_theme.menu_button_y, game_theme.menu_button);
if (!DIRECT_MODE) {
el_draw(el_menu_button);
}
return 0;
}
int game_restart(void)
{
char *og = curgame_dir;
game_save(-1);
game_done(0);
if (game_init(og)) {
game_error(og);
return 0;
}
return 0;
}
int static cur_vol = 0;
void free_last_music(void);
void game_stop_mus(int ms)
{
snd_stop_mus(ms);
free_last_music();
}
int game_change_vol(int d, int val)
{
int v = snd_volume_mus(-1);
int pc = snd_vol_to_pcn(v);
int opc = pc;
if (d) {
pc += d;
if (pc < 0)
pc = 0;
if (pc > 100)
pc = 100;
while (snd_vol_to_pcn(v) != pc)
v += (d<0)?-1:1;
} else {
v = val;
pc = snd_vol_to_pcn(v);
}
if (!pc)
v = 0;
snd_volume_mus(v);
if (opc && !pc) {
game_stop_mus(0);
}
if (!opc && pc) {
game_music_player();
}
cur_vol = snd_volume_mus(-1);
opt_vol = cur_vol;
return 0;
}
static void sounds_reload(void);
int game_change_hz(int hz)
{
if (!hz)
return -1;
snd_done();
free_last_music();
snd_init(hz);
snd_volume_mus(cur_vol);
snd_free_wav(game_theme.click);
game_theme.click = snd_load_wav(game_theme.click_name);
sounds_reload();
game_music_player();
opt_hz = snd_hz();
return 0;
}
unsigned int timer_counter = 0;
gtimer_t timer_han = NULL_TIMER;
static void anigif_do(void *data)
{
void *v;
img_t img;
if (browse_dialog || menu_shown || gfx_fading() || minimized())
return;
game_cursor(CURSOR_CLEAR);
if (gfx_frame_gif(el_img(el_spic))) { /* scene */
game_cursor(CURSOR_DRAW);
gfx_update_gif(el_img(el_spic));
}
game_cursor(CURSOR_CLEAR);
for (v = NULL; (img = txt_layout_images(txt_box_layout(el_box(el_scene)), &v)); ) { /* scene */
if ((img != el_img(el_spic)) && gfx_frame_gif(img)) {
game_cursor(CURSOR_DRAW);
gfx_update_gif(img);
}
}
game_cursor(CURSOR_CLEAR);
for (v = NULL; (img = txt_layout_images(txt_box_layout(el_box(el_inv)), &v)); ) { /* inv */
if (gfx_frame_gif(img)) {
game_cursor(CURSOR_DRAW);
gfx_update_gif(img);
}
}
game_cursor(CURSOR_CLEAR);
for (v = NULL; (img = txt_layout_images(el_layout(el_title), &v)); ) { /* title */
if (gfx_frame_gif(img)) {
game_cursor(CURSOR_DRAW);
gfx_update_gif(img);
}
}
game_cursor(CURSOR_CLEAR);
for (v = NULL; (img = txt_layout_images(el_layout(el_ways), &v)); ) { /* ways */
if (gfx_frame_gif(img)) {
game_cursor(CURSOR_DRAW);
gfx_update_gif(img);
}
}
game_cursor(CURSOR_ON);
}
int counter_fn(int interval, void *p)
{
timer_counter ++;
if (gfx_is_drawn_gifs() && !DIRECT_MODE)
push_user_event(anigif_do, NULL);
return interval;
}
int game_use_theme(void)
{
int rc = 0;
game_theme_changed = 0;
game_own_theme = 0;
game_theme.changed = CHANGED_ALL;
if (game_default_theme()) {
fprintf(stderr, "Can't load default theme.\n");
return -1;
}
if (curgame_dir && (!idf_access(game_idf, THEME_FILE) || !access(dirpath(THEME_FILE), R_OK))) {
game_own_theme = 1;
}
if (game_own_theme && opt_owntheme) {
theme_relative = 1;
rc = theme_load(THEME_FILE);
theme_relative = 0;
} else if (curtheme_dir && strlowcmp(DEFAULT_THEME, curtheme_dir)) {
rc = game_theme_load(curtheme_dir);
}
return rc;
}
int game_init(const char *name)
{
getdir(game_cwd, sizeof(game_cwd));
unix_path(game_cwd);
if (name)
game_err_msg(NULL);
snd_init(opt_hz);
game_change_vol(0, opt_vol);
if (game_select(name))
return -1;
if (game_theme_optimize())
return -1;
if (game_apply_theme()) {
game_theme_select(DEFAULT_THEME);
return -1;
}
timer_han = gfx_add_timer(HZ, counter_fn, NULL);
if (!curgame_dir) {
game_menu(menu_games);
} else {
if (!game_load(-1)) /* tmp save */
goto out;
if (opt_autosave && !game_load(0)) /* autosave */
goto out;
game_cmd("look");
custom_theme_warn();
if (opt_autosave)
game_save(0);
}
out:
return 0;
}
void free_last_music(void)
{
if (last_music)
free(last_music);
last_music = NULL;
}
void free_last(void)
{
if (last_pict)
free(last_pict);
if (last_title)
free(last_title);
if (last_cmd)
free(last_cmd);
last_pict = last_title = last_cmd = NULL;
game_stop_mus(500);
sounds_free();
}
void game_release_theme(void)
{
int i;
mouse_reset(1);
game_cursor(CURSOR_OFF);
if (el_img(el_spic))
gfx_free_image(el_img(el_spic));
for (i = 0; i < el_max; i++) {
struct el *o;
o = el(i);
if (o->type == elt_layout && o->p.p) {
txt_layout_free(o->p.lay);
} else if (o->type == elt_box && o->p.p) {
txt_layout_free(txt_box_layout(o->p.box));
txt_box_free(o->p.box);
}
o->p.p = NULL;
o->drawn = 0;
}
if (menu)
gfx_free_image(menu);
if (menubg)
gfx_free_image(menubg);
menu = menubg = NULL;
gfx_update(0, 0, game_theme.w, game_theme.h);
}
void game_done(int err)
{
gfx_del_timer(timer_han);
timer_han = NULL_TIMER;
if (opt_autosave && curgame_dir && !err)
game_save(0);
setdir(game_cwd);
// cfg_save();
if (menu_shown)
menu_toggle();
game_release_theme();
free_last();
game_theme_free();
input_clear();
snd_done();
instead_done();
//#ifndef ANDROID
// gfx_video_done();
//#endif
curgame_dir = NULL;
game_own_theme = 0;
idf_done(game_idf); game_idf = NULL;
// SDL_Quit();
}
static void el_size(int i, int *w, int *h)
{
int type;
type = el(i)->type;
if (type == elt_layout)
txt_layout_size(el_layout(i), w, h);
else if (type == elt_box)
txt_box_size(el_box(i), w, h);
else if (type == elt_image) {
if (w)
*w = gfx_img_w(el_img(i));
if (h)
*h = gfx_img_h(el_img(i));
}
}
#define el_clear(n) _el_clear(n, game_clear)
#define el_clear_nobg(n) _el_clear(n, NULL)
static int _el_clear(int n, clear_fn clear)
{
void *v;
img_t img;
int x, y, w, h;
struct el *o;
o = el(n);
if (!o || !o->drawn)
return 0;
x = o->x;
y = o->y;
el_size(n, &w, &h);
o->drawn = 0;
if (clear)
clear(x, y, w, h);
if (o->type == elt_box) {
for (v = NULL; (img = txt_layout_images(txt_box_layout(el_box(n)), &v)); )
gfx_dispose_gif(img);
} else if (o->type == elt_layout) {
for (v = NULL; (img = txt_layout_images(el_layout(n), &v)); )
gfx_dispose_gif(img);
} else if (o->type == elt_image)
gfx_dispose_gif(el_img(n));
return 1;
}
static void el_update(int n)
{
int x, y, w, h;
struct el *o;
o = el(n);
// if (!o->drawn)
// return;
x = o->x;
y = o->y;
el_size(n, &w, &h);
game_cursor(CURSOR_DRAW);
gfx_update(x, y, w, h);
return;
}
static void el_clip(int n)
{
int x, y, w, h;
struct el *o;
o = el(n);
x = o->x;
y = o->y;
el_size(n, &w, &h);
gfx_clip(x, y, w, h);
return;
}
int box_isscroll_up(int n)
{
if (el(n)->type != elt_box)
return -1;
if (txt_box_off(el_box(n)))
return 0;
return -1;
}
int box_isscroll_down(int n)
{
layout_t l;
int off;
int h, hh;
if (el(n)->type != elt_box)
return -1;
el_size(n, NULL, &h);
l = txt_box_layout(el_box(n));
txt_layout_real_size(l, NULL, &hh);
off = txt_box_off(el_box(n));
if (hh - off > h)
return 0;
return -1;
}
void box_update_scrollbar(int n)
{
struct el *elup = NULL;
struct el *eldown = NULL;
// struct el *elslide;
layout_t l;
int x1, y1;
int x2, y2;
int off;
int w, h, hh;
if (n == el_scene) {
elup = el(el_sup);
eldown = el(el_sdown);
x1 = game_theme.a_up_x;
y1 = game_theme.a_up_y;
x2 = game_theme.a_down_x;
y2 = game_theme.a_down_y;
// elslide = el(el_sslide);
} else if (n == el_inv) {
elup = el(el_iup);
eldown = el(el_idown);
x1 = game_theme.inv_a_up_x;
y1 = game_theme.inv_a_up_y;
x2 = game_theme.inv_a_down_x;
y2 = game_theme.inv_a_down_y;
// elslide = el(el_islide);
}
if (!elup || !eldown)
return;
// if (x1 == -1 || y1 == -1 || x2 == -1 || y2 == -1)
el_size(n, &w, &h);
if (x1 == -1)
x1 = el(n)->x + w + game_theme.pad;
if (y1 == -1)
y1 = el(n)->y;
if (x2 == -1)
x2 = x1;
if (y2 == -1)
y2 = y1 + h - gfx_img_h(game_theme.a_down);
l = txt_box_layout(el_box(n));
txt_layout_real_size(l, NULL, &hh);
off = txt_box_off(el_box(n));
if (el_clear(elup->id)) {
if (elup->x != x1 || elup->y != y1)
el_update(elup->id);
}
if (el_clear(eldown->id)) {
if (eldown->x != x2 || eldown->y != y2)
el_update(eldown->id);
}
elup->x = x1;
elup->y = y1;
eldown->x = x2;
eldown->y = y2;
el_clear(elup->id);
el_clear(eldown->id);
if (hh - off > h)
el_draw(eldown->id);
if (off)
el_draw(elup->id);
el_update(elup->id);
el_update(eldown->id);
}
void el_draw(int n)
{
int x, y;
struct el *o;
o = el(n);
x = o->x;
y = o->y;
if (!o->p.p)
return;
game_cursor(CURSOR_CLEAR);
if (o->type == elt_image)
gfx_draw(o->p.img, x, y);
else if (o->type == elt_layout)
txt_layout_draw(o->p.lay, x, y);
else if (o->type == elt_box) {
txt_box_draw(o->p.box, x, y);
box_update_scrollbar(o->id);
}
o->drawn = 1;
return;
}
img_t game_pict_scale(img_t img, int ww, int hh)
{
img_t img2 = img;
int w, h, www, hhh;
float scale1, scale2, scale = 1.0f;
game_pic_w = gfx_img_w(img);
game_pic_h = gfx_img_h(img);
if (!cache_have(gfx_image_cache(), img))
return img; /* do not scale sprites! */
if (game_theme.scale > 1.0f)
theme_img_scale(&img);
w = gfx_img_w(img);
h = gfx_img_h(img);
if (ww == -1)
ww = w;
if (hh == -1)
hh = h;
if (w <= ww && h <= hh)
return img;
www = ww;
hhh = hh;
while (scale * (float)w > ww || scale * (float)h > hh) {
scale1 = (float)(www - 2) / (float)(w);
scale2 = (float)(hhh - 2) / (float)(h);
scale = (scale1<scale2) ? scale1:scale2;
www -= 1;
hhh -= 1;
if (www <= 0 || hhh <=0)
break;
}
if (scale < 0)
scale = 0;
img2 = gfx_scale(img, scale, scale);
gfx_free_image(img);
return img2;
}
int game_menu_box_wh(const char *txt, int *w, int *h)
{
layout_t lay;
int b = game_theme.border_w;
int pad = game_theme.pad;
lay = txt_layout(game_theme.menu_font, ALIGN_CENTER, game_theme.w - 2 * (b + pad), 0);
txt_layout_set(lay, (char*)txt);
txt_layout_real_size(lay, w, h);
txt_layout_free(lay);
return 0;
}
void game_menu_box_width(int show, const char *txt, int width)
{
// img_t menu;
int w, h, mw, mh;
int x, y;
int b = game_theme.border_w;
int pad = game_theme.pad;
layout_t lay = NULL;
menu_shown = show;
el(el_menu)->drawn = 0;
if (el_layout(el_menu)) {
_txt_layout_free(el_layout(el_menu));
lay = el_layout(el_menu);
}
if (menubg) {
game_cursor(CURSOR_CLEAR);
gfx_draw(menubg, mx, my);
gfx_free_image(menubg);
menubg = NULL;
}
if (menu) {
gfx_free_image(menu);
menu = NULL;
}
if (!DIRECT_MODE) {
el_clear(el_menu_button);
if (!show)
el_draw(el_menu_button);
}
if (!show) {
game_cursor(CURSOR_DRAW);
gfx_flip();
return;
}
if (!lay) {
lay = txt_layout(game_theme.menu_font, ALIGN_CENTER, game_theme.w - 2 * (b + pad), 0);
txt_layout_color(lay, game_theme.menu_fg);
txt_layout_link_color(lay, game_theme.menu_link);
txt_layout_active_color(lay, game_theme.menu_alink);
txt_layout_font_height(lay, game_theme.menu_font_height);
}
else
txt_layout_set_size(lay, game_theme.w - 2 * (b + pad), 0);
txt_layout_set(lay, (char*)txt);
txt_layout_real_size(lay, &w, &h);
if (width)
w = width;
txt_layout_set_size(lay, w, h);
txt_layout_set(lay, (char*)txt);
menu = gfx_new(w + (b + pad)*2, h + (b + pad)*2);
gfx_img_fill(menu, 0, 0, w + (b + pad)*2, h + (b + pad)*2, game_theme.border_col);
gfx_img_fill(menu, b, b, w + pad*2, h + pad*2, game_theme.menu_bg);
gfx_set_alpha(menu, game_theme.menu_alpha);
x = (game_theme.w - w)/2;
y = (game_theme.h - h)/2;
mx = x - b - pad;
my = y - b - pad;
mw = w + (b + pad) * 2;
mh = h + (b + pad) * 2;
game_cursor(CURSOR_CLEAR);
menubg = gfx_grab_screen(mx, my, mw, mh);
gfx_draw(menu, mx, my);
el_set(el_menu, elt_layout, /*game_theme.win_x*/ x, y, lay);
el_draw(el_menu);
game_cursor(CURSOR_DRAW);
gfx_flip();
}
void game_menu_box(int show, const char *txt)
{
int w = 0;
if (cur_menu == menu_games) { /* hack a bit :( */
w = games_menu_maxw();
game_menu_gen();
} else if (cur_menu == menu_themes) {
w = themes_menu_maxw();
game_menu_gen();
}
return game_menu_box_width(show, txt, w);
}
int check_new_place(char *title)
{
int rc = 0;
if (!title && !last_title)
return 0;
if (!title && last_title) {
rc = 1;
} else if (!last_title || strcmp(title, last_title)) {
rc = 1;
}
if (last_title) {
free(last_title);
}
last_title = title;
return rc;
}
int check_new_pict(char *pict)
{
int rc = 0;
if (!pict && !last_pict)
return 0;
if (!pict && last_pict) {
rc = 1;
} else if (!last_pict || strcmp(pict, last_pict)) {
rc = 1;
}
if (last_pict) {
free(last_pict);
}
last_pict = pict;
return rc;
}
static int check_fading(int *new_scene)
{
int rc;
int st;
instead_function("instead.get_fading", NULL);
rc = instead_bretval(0);
st = instead_iretval(1);
if (st < 0)
st = 0;
else if (st > 255)
st = 255;
instead_clear();
if (new_scene)
*new_scene = rc;
return st;
}
static void game_autosave(void)
{
int b,r;
if (!curgame_dir)
return;
instead_function("instead.get_autosave", NULL);
b = instead_bretval(0);
r = instead_iretval(1);
instead_clear();
if (b) {
r = r % MAX_SAVE_SLOTS;
game_save(r);
/* instead_eval("game.autosave = false;"); instead_clear();*/
}
}
static void dec_music(void *data)
{
int rc;
if (!curgame_dir)
return;
instead_function("instead.dec_music_loop", NULL);
rc = instead_iretval(0);
instead_clear();
if (rc == -1)
free_last_music();
snd_volume_mus(cur_vol); /* reset volume */
}
void game_music_finished(void)
{
push_user_event(&dec_music, NULL);
}
void game_music_player(void)
{
int loop;
char *mus;
if (!snd_volume_mus(-1))
return;
if (!opt_music || !curgame_dir)
return;
instead_function("instead.get_music", NULL);
mus = instead_retval(0);
loop = instead_iretval(1);
unix_path(mus);
instead_clear();
if (mus && loop == -1) { /* disabled, 0 - forever, 1-n - loops */
free(mus);
mus = NULL;
}
if (!mus) {
if (last_music) {
game_stop_mus(500);
}
} else if (!last_music && mus) {
game_stop_mus(500);
last_music = mus;
snd_play_mus(mus, 0, loop - 1);
} else if (strcmp(last_music, mus)) {
game_stop_mus(500);
last_music = mus;
snd_play_mus(mus, 0, loop - 1);
} else
free(mus);
}
#define MAX_WAVS SND_CHANNELS * 2
static LIST_HEAD(sounds);
static int sounds_nr = 0;
typedef struct {
struct list_head list;
char *fname;
wav_t wav;
int loaded;
} _snd_t;
static _snd_t *channels[SND_CHANNELS] = {};
void game_channel_finished(int channel)
{
// fprintf(stderr, "finished: %d\n", channel);
channels[channel % SND_CHANNELS] = NULL;
}
static int sound_playing(_snd_t *snd)
{
int i;
for (i = 0; i < SND_CHANNELS; i++) {
if (channels[i] == snd)
return i;
}
return -1;
}
const char *sound_channel(int i)
{
_snd_t *sn;
sn = channels[i];
if (!sn)
return NULL;
return sn->fname;
}
static void sound_free(_snd_t *sn)
{
if (!sn)
return;
free(sn->fname);
snd_free_wav(sn->wav);
list_del(&sn->list);
free(sn);
sounds_nr --;
}
static void sounds_shrink(void)
{
struct list_head *pos, *pos2;
_snd_t *sn;
pos = sounds.next;
// fprintf(stderr,"shrink try\n");
while (pos != &sounds && sounds_nr > MAX_WAVS) {
sn = (_snd_t*)pos;
if (sound_playing(sn) != -1 || sn->loaded) {
pos = pos->next;
continue;
}
pos2 = pos->next;
sound_free(sn);
pos = pos2;
// fprintf(stderr,"shrink by 1\n");
}
}
void sounds_free(void)
{
int i = 0;
snd_halt_chan(-1, 0); /* halt sounds */
while (!list_empty(&sounds)) {
_snd_t *sn = (_snd_t*)(sounds.next);
// if (sound_playing(sn) == -1) {
sound_free(sn);
// }
}
for (i = 0; i < SND_CHANNELS; i++)
channels[i] = NULL;
sounds_nr = 0;
}
static _snd_t *sound_find(const char *fname)
{
struct list_head *pos;
_snd_t *sn;
list_for_each(pos, &sounds) {
sn = (_snd_t*)pos;
if (!strcmp(fname, sn->fname)) {
list_move(&sn->list, &sounds); // move it on head
return sn;
}
}
return NULL;
}
static void sound_play(_snd_t *sn, int chan, int loop)
{
int c = snd_play(sn->wav, chan, loop - 1);
// fprintf(stderr, "added: %d\n", c);
if (c == -1)
return;
channels[c] = sn;
}
static _snd_t *sound_add(const char *fname)
{
wav_t w;
_snd_t *sn;
if (!fname || !*fname)
return NULL;
sn = malloc(sizeof(_snd_t));
if (!sn)
return NULL;
INIT_LIST_HEAD(&sn->list);
sn->fname = strdup(fname);
sn->loaded = 0;
if (!sn->fname) {
free(sn);
return NULL;
}
w = snd_load_wav(fname);
if (!w)
goto err;
sn->wav = w;
sounds_shrink();
list_add(&sn->list, &sounds);
sounds_nr ++;
return sn;
err:
free(sn->fname);
free(sn);
return NULL;
}
static void sounds_reload(void)
{
struct list_head *pos;
_snd_t *sn;
list_for_each(pos, &sounds) {
sn = (_snd_t*)pos;
snd_free_wav(sn->wav);
sn->wav = snd_load_wav(sn->fname);
}
}
int sound_load(const char *fname)
{
_snd_t *sn;
sn = sound_find(fname);
if (sn) {
sn->loaded ++; /* to pin */
return 0;
}
sn = sound_add(fname);
if (!sn)
return -1;
sn->loaded = 1;
return 0;
}
void sound_unload(const char *fname)
{
_snd_t *sn;
sn = sound_find(fname);
if (!sn || !sn->loaded)
return;
sn->loaded --;
if (!sn->loaded && sound_playing(sn) == -1)
sound_free(sn);
return;
}
static int _play_combined_snd(char *filename, int chan, int loop)
{
char *str;
char *p, *ep;
_snd_t *sn;
p = str = strdup(filename);
if (!str)
return -1;
p = strip(p);
while (*p) {
int c = chan, l = loop;
int at = 0;
ep = p + strcspn(p, ";@");
if (*ep == '@') {
at = 1;
*ep = 0; ep ++;
sscanf(ep, "%d,%d", &c, &l);
ep += strcspn(ep, ";");
if (*ep)
ep ++;
} else if (*ep == ';') {
*ep = 0; ep ++;
} else if (*ep) {
goto err;
}
p = strip(p);
sn = sound_find(p);
if (!sn)
sn = sound_add(p);
if (sn)
sound_play(sn, c, l);
else if (at || c != -1) /* if @ or specific channel */
snd_halt_chan(c, 500);
p = ep;
}
free(str);
return 0;
err:
free(str);
return -1;
}
void game_sound_player(void)
{
char *snd;
int chan = -1;
int loop = 1;
struct instead_args args[] = {
{ .val = "nil", .type = INSTEAD_NIL },
{ .val = "-1", .type = INSTEAD_NUM },
{ .val = NULL }
};
if (!snd_volume_mus(-1))
return;
instead_function("instead.get_sound", NULL);
loop = instead_iretval(2);
chan = instead_iretval(1);
snd = instead_retval(0);
instead_clear();
if (!snd) {
if (chan != -1) {
/* halt channel */
snd_halt_chan(chan, 500);
instead_function("instead.set_sound", args); instead_clear();
}
return;
}
instead_function("instead.set_sound", args); instead_clear();
unix_path(snd);
_play_combined_snd(snd, chan, loop);
free(snd);
}
static char *get_inv(void)
{
char *ni;
struct instead_args args[] = {
{ .val = NULL, .type = INSTEAD_BOOL },
{ .val = NULL, },
};
args[0].val = (INV_MODE(game_theme.inv_mode) == INV_MODE_HORIZ)?"true":"false";
instead_function("instead.get_inv", args);
ni = instead_retval(0);
instead_clear();
return ni;
}
static int find_diff_pos(const char *p1, const char *p2)
{
int pos = 0;
if (!p1 || !p2)
return -1;
while ((*p1 == *p2) && *p1) {
p1 ++;
p2 ++;
pos ++;
}
if (!*p1)
return -1;
return pos;
}
static void scroll_to_diff(const char *cmdstr, int cur_off)
{
int off = 0;
int pos = 0;
int h = 0;
int hh = 0;
pos = find_diff_pos(cmdstr, last_cmd);
if (pos == -1)
off = cur_off;
else
off = txt_layout_pos2off(txt_box_layout(el_box(el_scene)), pos, &hh);
el_size(el_scene, NULL, &h);
if (cur_off <= off && cur_off + h >= off + hh) { /* do not scroll */
off = cur_off;
}
txt_box_scroll(el_box(el_scene), off);
}
int game_highlight(int x, int y, int on);
int game_pict_modify(img_t p)
{
static int modify = 0;
int last = modify;
if (p && ((el_img(el_spic) == p) || p == gfx_screen(NULL)))
modify = 1;
else
modify = 0;
return last;
}
static void game_pict_clip(void)
{
int x, y, w, h;
if (game_theme.gfx_mode == GFX_MODE_EMBEDDED) {
el_clip(el_scene);
return;
}
if (game_theme.gfx_mode != GFX_MODE_FLOAT) {
x = game_theme.win_x;
y = game_theme.win_y;
w = game_theme.win_w;
h = game_theme.win_h;
} else {
x = game_theme.gfx_x;
y = game_theme.gfx_y;
w = game_theme.max_scene_w;
h = game_theme.max_scene_h;
}
gfx_clip(x, y, w, h);
}
static void game_redraw_pic(void)
{
int x, y, ox, oy;
if (game_pict_coord(&x, &y, NULL, NULL))
return;
game_pict_clip();
ox = el(el_spic)->x;
oy = el(el_spic)->y;
el(el_spic)->x = x;
el(el_spic)->y = y;
el_clear(el_spic);
el_draw(el_spic);
gfx_noclip();
el_update(el_spic);
el(el_spic)->x = ox;
el(el_spic)->y = oy;
}
int game_cmd(char *cmd)
{
int old_off;
int fading = 0;
int new_pict = 0;
int new_place = 0;
int redraw_pict = 0;
int title_h = 0, ways_h = 0, pict_h = 0;
char buf[1024];
char *cmdstr = NULL;
char *invstr = NULL;
char *waystr = NULL;
char *title = NULL;
char *pict = NULL;
img_t oldscreen = NULL;
int dd = (DIRECT_MODE);
int rc = 0;
int new_scene = 0;
if (menu_shown)
return -1;
// if (dd)
game_cursor(CURSOR_CLEAR);
cmdstr = instead_cmd(cmd); rc = !instead_bretval(1); instead_clear();
game_music_player();
game_sound_player();
if (DIRECT_MODE) {
if (cmdstr)
free(cmdstr);
if (game_theme_changed == 2) { /* cursor change only? */
img_t offscreen = gfx_new(game_theme.w, game_theme.h);
oldscreen = gfx_screen(offscreen);
gfx_draw(oldscreen, 0, 0);
game_theme_update();
game_theme_changed = 1;
offscreen = gfx_screen(oldscreen);
gfx_change_screen(offscreen, 1);
gfx_free_image(offscreen);
}
if (game_pict_modify(NULL))
goto out;
return 0;
} else if (dd) { /* disable direct mode on the fly */
game_theme_changed = 2; /* force redraw */
game_cursor(CURSOR_DRAW);
}
if (!cmdstr) {
if (game_pict_modify(NULL)) /* redraw pic only */
game_redraw_pic();
if (!rc)
goto inv; /* hackish? ok, yes it is... */
goto err; /* really nothing to do */
}
fading = check_fading(&new_scene);
instead_function("instead.get_title", NULL);
title = instead_retval(0);
instead_clear();
new_place = check_new_place(title);
instead_function("instead.get_picture", NULL);
pict = instead_retval(0);
instead_clear();
unix_path(pict);
new_pict = check_new_pict(pict);
if (game_theme_changed == 2 && opt_owntheme && !fading)
fading = 1; /* one frame at least */
if (fading) { /* take old screen */
game_cursor(CURSOR_CLEAR);
img_t offscreen = gfx_new(game_theme.w, game_theme.h);
oldscreen = gfx_screen(offscreen);
gfx_draw(oldscreen, 0, 0);
}
if (game_theme_changed == 2 && opt_owntheme) {
game_theme_update();
game_theme_changed = 1;
new_place = 1;
if (pict)
new_pict = 1;
}
if (new_place)
el_clear(el_title);
if (title && *title) {
snprintf(buf, sizeof(buf), "<a:look>%s</a>", title);
txt_layout_set(el_layout(el_title), buf);
txt_layout_size(el_layout(el_title), NULL, &title_h);
title_h += game_theme.font_size * game_theme.font_height / 2; // todo?
} else
txt_layout_set(el_layout(el_title), NULL);
if (new_pict || fading || game_pict_modify(NULL) ||
(new_place && (game_theme.gfx_mode == GFX_MODE_FIXED))) {
redraw_pict = 1;
}
game_pict_clip();
if (redraw_pict) {
if (el_img(el_spic)) {
if (game_theme.gfx_mode == GFX_MODE_EMBEDDED)
el_clear_nobg(el_spic);
else
el_clear(el_spic);
if (new_pict) {
gfx_free_image(el_img(el_spic));
el(el_spic)->p.p = NULL;
}
}
}
if (pict) {
int w, h, x;
img_t img;
if (new_pict) {
img = gfx_load_image(pict);
if (el_img(el_spic))
gfx_free_image(el_img(el_spic));
el(el_spic)->p.p = NULL;
if (game_theme.gfx_mode != GFX_MODE_FLOAT)
img = game_pict_scale(img, game_theme.win_w, game_theme.max_scene_h);
else
img = game_pict_scale(img, game_theme.max_scene_w, game_theme.max_scene_h);
} else
img = el_img(el_spic);
if (img) {
w = gfx_img_w(img);
h = gfx_img_h(img);
if (game_theme.gfx_mode != GFX_MODE_FLOAT) {
x = (game_theme.win_w - w)/2 + game_theme.win_x;
if (redraw_pict)
el_set(el_spic, elt_image, x, game_theme.win_y + title_h, img);
} else {
x = (game_theme.max_scene_w - w)/2 + game_theme.gfx_x;
if (redraw_pict)
el_set(el_spic, elt_image, x, game_theme.gfx_y/* + (game_theme.max_scene_h - h)/2*/, img);
}
pict_h = h;
}
}
gfx_noclip();
/* clear area */
el_clear(el_ways);
el_clear(el_scene);
instead_function("instead.get_ways", NULL);
waystr = instead_retval(0);
instead_clear();
if (waystr) {
int l = strlen(waystr);
if (l && waystr[l - 1] == '\n')
waystr[l - 1] = 0;
}
if (game_theme.gfx_mode != GFX_MODE_EMBEDDED) {
txt_layout_set(el_layout(el_ways), waystr);
txt_layout_size(el_layout(el_ways), NULL, &ways_h);
}
old_off = txt_box_off(el_box(el_scene));
if (game_theme.gfx_mode == GFX_MODE_EMBEDDED) {
char *p;
pict_h = 0; /* to fake code bellow */
txt_layout_set(txt_box_layout(el_box(el_scene)), ""); /* hack, to null layout, but not images */
if (el_img(el_spic)) {
txt_layout_add_img(txt_box_layout(el_box(el_scene)),"scene", el_img(el_spic));
txt_layout_add(txt_box_layout(el_box(el_scene)), "<c><g:scene></c>\n");
}
p = malloc(strlen(cmdstr) + ((waystr)?strlen(waystr):0) + 16);
if (p) {
*p = 0;
if ((waystr && *waystr) || el_img(el_spic)) { /* is this hack needed? */
if (waystr)
strcpy(p, waystr);
strcat(p, "\n"); /* small hack */
}
strcat(p, cmdstr);
free(cmdstr);
cmdstr = p;
} else { /* paranoia? Yes... */
txt_layout_add(txt_box_layout(el_box(el_scene)), waystr);
txt_layout_add(txt_box_layout(el_box(el_scene)), "<l></l>\n"); /* small hack */
}
txt_layout_add(txt_box_layout(el_box(el_scene)), cmdstr);
txt_box_set(el_box(el_scene), txt_box_layout(el_box(el_scene)));
} else {
if (game_theme.gfx_mode == GFX_MODE_FLOAT)
pict_h = 0;
txt_layout_set(txt_box_layout(el_box(el_scene)), cmdstr);
txt_box_set(el_box(el_scene), txt_box_layout(el_box(el_scene)));
}
if (!new_scene)
scroll_to_diff(cmdstr, old_off);
FREE(last_cmd);
last_cmd = cmdstr;
el(el_ways)->y = el(el_title)->y + title_h + pict_h;
if (waystr)
free(waystr);
el(el_scene)->y = el(el_ways)->y + ways_h;
/*
game_clear(game_theme.win_x, game_theme.win_y + pict_h + title_h,
game_theme.win_w, game_theme.win_h - pict_h - title_h); */
/* draw title and ways */
if (new_place)
el_draw(el_title);
if (game_theme.gfx_mode != GFX_MODE_EMBEDDED) {
el_draw(el_ways);
if (redraw_pict) {
game_pict_clip();
el_draw(el_spic);
gfx_noclip();
}
}
txt_box_resize(el_box(el_scene), game_theme.win_w, game_theme.win_h - title_h - ways_h - pict_h);
el_draw(el_scene);
inv:
if (inv_enabled()) {
int off;
invstr = get_inv();
off = txt_box_off(el_box(el_inv));
txt_layout_set(txt_box_layout(el_box(el_inv)), invstr);
txt_box_set(el_box(el_inv), txt_box_layout(el_box(el_inv)));
txt_box_scroll(el_box(el_inv), off);
if (invstr)
free(invstr);
el_clear(el_inv);
el_draw(el_inv);
}
if (fading) {
img_t offscreen;
game_cursor(CURSOR_CLEAR);
gfx_stop_gif(el_img(el_spic));
offscreen = gfx_screen(oldscreen);
gfx_change_screen(offscreen, fading);
gfx_start_gif(el_img(el_spic));
gfx_free_image(offscreen);
// input_clear();
}
{ /* highlight new scene, to avoid flickering */
int x, y;
gfx_cursor(&x, &y);
game_highlight(x, y, 1);
}
if (fading)
goto err;
out:
game_cursor(CURSOR_DRAW);
gfx_flip();
// input_clear();
err:
game_autosave();
#if 0
if (err_msg) {
mouse_reset(1);
game_menu(menu_warning);
return -1;
}
#endif
return rc;
}
void game_update(int x, int y, int w, int h)
{
game_cursor(CURSOR_DRAW);
gfx_update(x, y, w, h);
}
void game_xref_update(xref_t xref, int x, int y)
{
game_cursor(CURSOR_CLEAR);
xref_update(xref, x, y, game_clear, game_update);
}
xref_t use_xref = NULL;
int disable_use(void)
{
if (use_xref) {
xref_set_active(use_xref, 0);
if (xref_layout(use_xref) == txt_box_layout(el_box(el_inv)))
game_xref_update(use_xref, el(el_inv)->x, el(el_inv)->y);
else
game_xref_update(use_xref, el(el_scene)->x, el(el_scene)->y);
use_xref = NULL;
return 1;
}
return 0;
}
void enable_use(xref_t xref)
{
use_xref = xref;
xref_set_active(xref, 1);
if (xref_layout(use_xref) == txt_box_layout(el_box(el_inv)))
game_xref_update(use_xref, el(el_inv)->x, el(el_inv)->y);
else
game_xref_update(use_xref, el(el_scene)->x, el(el_scene)->y);
}
struct el *look_obj(int x, int y)
{
int i;
for (i = 0; i < el_max; i++) {
int w, h;
if (el(i)->drawn && el(i)->id == el_menu) {
return el(i);
}
if (x < el(i)->x || y < el(i)->y || !el(i)->drawn)
continue;
el_size(i, &w, &h);
if (x >= el(i)->x && y >= el(i)->y && x <= el(i)->x + w && y <= el(i)->y + h)
return el(i);
}
return NULL;
}
static xref_t get_nearest_xref(int i, int mx, int my);
xref_t look_xref(int x, int y, struct el **elem)
{
struct el *o;
int type;
xref_t xref = NULL;
o = look_obj(x, y);
if (elem)
*elem = o;
if (!o)
return NULL;
type = o->type;
if (type == elt_layout)
xref = txt_layout_xref(o->p.lay, x - o->x, y - o->y);
else if (type == elt_box)
xref = txt_box_xref(o->p.box, x - o->x, y - o->y);
#ifdef ANDROID
if (!xref) {
int xc, yc, r;
xref = get_nearest_xref(o->id, x, y);
if (!xref)
return NULL;
r = fnt_height(txt_layout_font(xref_layout(xref))) * 2; /* radius is here */
if (!xref_position(xref, &xc, &yc)) {
if (o->type == elt_box && yc)
yc -= txt_box_off(el_box(o->id));
xc += o->x;
yc += o->y;
if (((x - xc)*(x - xc) + (y - yc)*(y - yc)) < (r * r))
return xref;
}
return NULL;
}
#endif
return xref;
}
static xref_t old_xref = NULL;
static struct el *old_el = NULL;
static int click_x = -1;
static int click_y = -1;
int game_paused(void)
{
return browse_dialog || menu_shown || use_xref || old_xref || gfx_fading() || minimized();
}
void menu_update(struct el *elem)
{
gfx_draw(menubg, mx, my);
gfx_draw(menu, mx, my);
txt_layout_draw(elem->p.lay, elem->x, elem->y);
gfx_update(mx, my, gfx_img_w(menu), gfx_img_h(menu));
// gfx_fill(x, y, w, h, game_theme.menu_bg);
}
int game_highlight(int x, int y, int on)
{
struct el *elem = NULL;
static struct el *oel = NULL;
static xref_t hxref = NULL;
xref_t xref = NULL;
if (on == 1) {
xref = look_xref(x, y, &elem);
if (xref && opt_hl && !xref_get_active(xref)) {
xref_set_active(xref, 1);
game_xref_update(xref, elem->x, elem->y);
}
}
if (hxref != xref && oel) {
if (hxref != use_xref && xref_get_active(hxref)) {
xref_set_active(hxref, 0);
if (on != -1)
game_xref_update(hxref, oel->x, oel->y);
}
hxref = NULL;
}
hxref = xref;
oel = elem;
return 0;
}
void mouse_reset(int hl)
{
if (hl)
game_highlight(-1, -1, 0);
else
game_highlight(-1, -1, -1);
disable_use();
motion_mode = 0;
old_xref = old_el = NULL;
}
void menu_toggle(void)
{
mouse_reset(1);
menu_shown ^= 1;
if (!menu_shown)
cur_menu = menu_main;
game_menu_box(menu_shown, game_menu_gen());
}
static int scroll_pup(int id)
{
if (box_isscroll_up(id))
return -1;
// game_highlight(-1, -1, 0);
if (game_theme.gfx_mode == GFX_MODE_EMBEDDED) {
int hh;
el_size(el_scene, NULL, &hh);
txt_box_scroll(el_box(id), -hh);
} else
txt_box_prev(el_box(id));
el_clear(id);
el_draw(id);
el_update(id);
return 0;
}
static int scroll_pdown(int id)
{
if (box_isscroll_down(id))
return -1;
// game_highlight(-1, -1, 0);
if (game_theme.gfx_mode == GFX_MODE_EMBEDDED) {
int hh;
el_size(el_scene, NULL, &hh);
txt_box_scroll(el_box(id), hh);
} else
txt_box_next(el_box(id));
el_clear(id);
el_draw(id);
el_update(id);
return 0;
}
int mouse_filter(int filter)
{
static unsigned int old_counter = 0;
if (!opt_filter)
return 0;
if (filter && (abs(old_counter - timer_counter) <= (400 / HZ))) /* 400 ms */
return -1;
old_counter = timer_counter;
return 0;
}
/* action: 0 - first click,1 - second, -1 - restore */
int game_click(int x, int y, int action, int filter)
{
int menu_mode = 0;
int use_mode = 0;
int go_mode = 0;
struct el *elem = NULL;
char buf[1024];
xref_t xref = NULL;
char *xref_txt;
if (!action) {
click_x = x;
click_y = y;
motion_y = y;
} else if (action == 1) {
click_x = -1;
click_y = -1;
}
if (action)
motion_mode = 0;
if (filter && opt_filter && action == 1) {
xref_t new_xref;
struct el *new_elem;
new_xref = look_xref(x, y, &new_elem);
if (new_xref != old_xref || new_elem != old_el) {
old_el = NULL;
if (old_xref) {
old_xref = NULL;
return 0; /* just filtered */
}
old_xref = NULL;
}
}
if (action == 1) {
xref = old_xref;
elem = old_el;
old_xref = NULL;
old_el = NULL;
} else { /* just press */
xref = look_xref(x, y, &elem);
if (xref) {
xref_set_active(xref, 1);
game_xref_update(xref, elem->x, elem->y);
} else if (elem && elem->type == elt_box && opt_motion &&
(!box_isscroll_up(elem->id) || !box_isscroll_down(elem->id))) {
motion_mode = 1;
motion_id = elem->id;
return 0;
}
old_xref = xref;
old_el = elem;
return 0;
}
/* now look only second press */
if (!xref) {
if (elem) {
if (elem->id == el_menu_button) {
menu_toggle();
} else if (elem->id == el_sdown) {
scroll_pdown(el_scene);
} else if (elem->id == el_sup) {
scroll_pup(el_scene);
} else if (elem->id == el_idown) {
scroll_pdown(el_inv);
} else if (elem->id == el_iup) {
scroll_pup(el_inv);
} else if (disable_use());
// el_update(el_inv);
motion_mode = 0;
} else if (disable_use()) {
// el_update(el_inv);
// gfx_flip();
}
return 0;
}
/* second xref */
if (elem->id == el_menu) {
// xref_set_active(xref, 0);
// txt_layout_update_links(elem->p.lay, elem->x, elem->y, game_clear);
if (game_menu_act(xref_get_text(xref))) {
return -1;
}
// game_menu_box(menu_shown, game_menu_gen());
// gfx_flip();
return 1;
}
xref_txt = xref_get_text(xref);
if (!strncmp("act ", xref_get_text(xref), 4)) {
menu_mode = 1;
xref_txt += 4;
} else if (!strncmp("use ", xref_get_text(xref), 4)) {
use_mode = 1;
xref_txt += 4;
} else if (!strncmp("go ", xref_get_text(xref), 3)) {
go_mode = 1;
xref_txt += 3;
} else if (elem->id == el_inv) {
use_mode = 1;
}
if (!use_xref) {
if (use_mode) {
enable_use(xref);
// el_update(el_inv);
return 0;
}
if (menu_mode) {
if (elem->id == el_inv)
snprintf(buf, sizeof(buf), "use %s", xref_txt);
else
snprintf(buf, sizeof(buf), "act %s", xref_txt);
} else
snprintf(buf, sizeof(buf), "%s", xref_get_text(xref));
if (mouse_filter(filter))
return 0;
if (opt_click)
snd_play(game_theme.click, -1, 0);
game_cmd(buf);
return 1;
}
if (menu_mode || go_mode || elem->id == el_title)
return 0;
if (use_xref == xref)
snprintf(buf,sizeof(buf), "use %s", xref_txt);
else {
if (!strncmp("use ", xref_get_text(use_xref), 4)) /* already use */
snprintf(buf,sizeof(buf), "%s,%s", xref_get_text(use_xref), xref_txt);
else
snprintf(buf,sizeof(buf), "use %s,%s", xref_get_text(use_xref), xref_txt);
}
if (mouse_filter(filter))
return 0;
disable_use();
if (opt_click)
snd_play(game_theme.click, -1, 0);
game_cmd(buf);
return 1;
}
void mouse_restore(void)
{
game_click(click_x, click_y, -1, 0);
}
void game_cursor(int on)
{
static img_t grab = NULL;
static img_t cur = NULL;
static int xc = 0, yc = 0, w = 0, h = 0; //, w, h;
int xx, yy, ww, hh;
gfx_getclip(&xx, &yy, &ww, &hh);
gfx_noclip();
if (on == CURSOR_OFF)
cur = NULL;
if (grab) {
gfx_draw(grab, xc, yc);
gfx_free_image(grab);
grab = NULL;
}
if (on == CURSOR_OFF) {
gfx_update(xc, yc, w, h);
goto out;
}
if (on == CURSOR_CLEAR)
goto out;
if (on != CURSOR_DRAW)
cur = (use_xref) ? game_theme.use:game_theme.cursor;
if (!cur)
goto out;
do {
int ox = xc;
int oy = yc;
int ow = w;
int oh = h;
if (on != CURSOR_DRAW) {
gfx_cursor(&xc, &yc);
xc -= game_theme.cur_x;
yc -= game_theme.cur_y;
}
w = gfx_img_w(cur);
h = gfx_img_h(cur);
grab = gfx_grab_screen(xc, yc, w, h);
if (mouse_focus())
gfx_draw(cur, xc, yc);
if (on != CURSOR_DRAW) {
gfx_update(xc, yc, w, h);
gfx_update(ox, oy, ow, oh);
}
} while (0);
out:
gfx_clip(xx, yy, ww, hh);
return;
}
static void scroll_up(int id, int count)
{
// int i;
if (box_isscroll_up(id))
return;
// game_highlight(-1, -1, 0);
// if (game_theme.gfx_mode == GFX_MODE_EMBEDDED)
txt_box_scroll(el_box(id), -(FONT_SZ(game_theme.font_size)) * count);
// else
// for (i = 0; i < count; i++)
// txt_box_prev_line(el_box(id));
el_clear(id);
el_draw(id);
el_update(id);
}
static void scroll_down(int id, int count)
{
// int i;
if (box_isscroll_down(id))
return;
// game_highlight(-1, -1, 0);
// if (game_theme.gfx_mode == GFX_MODE_EMBEDDED)
txt_box_scroll(el_box(id), (FONT_SZ(game_theme.font_size)) * count);
// else
// for (i = 0; i < count; i++)
// txt_box_next_line(el_box(id));
el_clear(id);
el_draw(id);
el_update(id);
}
static void scroll_motion(int id, int off)
{
game_highlight(-1, -1, 0);
if (!off || (off > 0 && box_isscroll_down(id)) ||
(off < 0 && box_isscroll_up(id)))
return;
txt_box_scroll(el_box(id), off);
el_clear(id);
el_draw(id);
el_update(id);
}
static int sel_el = 0;
static void frame_next(void)
{
if (sel_el == el_scene && !inv_enabled())
sel_el = el_inv;
switch(sel_el) {
default:
case 0:
sel_el = el_scene;
break;
case el_ways:
sel_el = el_scene;
break;
case el_scene:
sel_el = el_inv;
break;
case el_inv:
if (game_theme.gfx_mode != GFX_MODE_EMBEDDED &&
txt_layout_xrefs(el_layout(el_ways)))
sel_el = el_ways;
else
sel_el = el_scene;
break;
}
}
static void frame_prev(void)
{
switch(sel_el) {
default:
case 0:
sel_el = el_inv;
break;
case el_title:
sel_el = el_inv;
break;
case el_ways:
sel_el = el_inv;
break;
case el_scene:
if (game_theme.gfx_mode != GFX_MODE_EMBEDDED &&
txt_layout_xrefs(el_layout(el_ways)))
sel_el = el_ways;
else
sel_el = el_inv;
break;
case el_inv:
sel_el = el_scene;
break;
}
if (sel_el == el_inv && !inv_enabled())
sel_el = el_scene;
}
static int select_ref(int prev, int last);
static xref_t get_xref(int i, int last);
static void xref_jump(xref_t xref, struct el* elem);
static void select_frame(int prev)
{
struct el *elem = NULL;
int x, y, w, h;
gfx_cursor(&x, &y);
elem = look_obj(x, y);
if (elem)
sel_el = elem->id;
el(sel_el)->mx = x;
el(sel_el)->my = y;
if (menu_shown) {
sel_el = el_menu;
} else {
// int old_sel;
// if (!sel_el)
// frame_next();
// old_sel = sel_el;
// do {
if (prev) {
frame_prev();
} else {
frame_next();
}
// } while (!get_xref(sel_el, 0) && sel_el != old_sel);
}
el_size(sel_el, &w, &h);
x = el(sel_el)->mx;
y = el(sel_el)->my;
if (x < el(sel_el)->x || y < el(sel_el)->y ||
x > el(sel_el)->x + w || y > el(sel_el)->y + h) {
x = el(sel_el)->x + w / 2;
y = el(sel_el)->y + h / 2;
}
gfx_warp_cursor(x, y);
if (!look_xref(x, y, &elem) && elem) {
xref_t xref = get_nearest_xref(elem->id, x, y);
xref_jump(xref, elem);
}
}
static int xref_rel_position(xref_t xref, struct el *elem, int *x, int *y)
{
int rc = xref_position(xref, x, y);
if (!rc && elem->type == elt_box && y) {
*y -= txt_box_off(el_box(elem->id));
}
return rc;
}
static int xref_visible(xref_t xref, struct el *elem)
{
int x, y, w, h;
if (!elem || !xref)
return -1;
if (xref_rel_position(xref, elem, &x, &y))
return -1;
el_size(elem->id, &w, &h);
if (y < 0 || y >= h)
return -1;
return 0;
}
static xref_t get_nearest_xref(int i, int mx, int my)
{
xref_t xref = NULL;
xref_t min_xref = NULL;
int min_disp = game_theme.h * game_theme.h + game_theme.w * game_theme.w;
if (!i)
return NULL;
for (xref = get_xref(i, 0); !xref_visible(xref, el(i)); xref = xref_next(xref)) {
int x, y, disp;
if (xref_rel_position(xref, el(i), &x, &y))
continue;
disp = (x + el(i)->x - mx) * (x + el(i)->x - mx) + (y + el(i)->y - my) * (y + el(i)->y - my);
if (disp < min_disp) {
min_disp = disp;
min_xref = xref;
}
}
return min_xref;
}
static xref_t get_xref(int i, int last)
{
xref_t xref = NULL;
int type;
type = el(i)->type;
if (type == elt_layout) {
xref = txt_layout_xrefs(el_layout(i));
while (last && xref && xref_next(xref))
xref = xref_next(xref);
} else if (type == elt_box) {
xref = txt_box_xrefs(el_box(i));
while (!last && xref && !xref_visible(xref_prev(xref), el(i))) /* try find visible one */
xref = xref_prev(xref);
while (last && xref && !xref_visible(xref_next(xref), el(i)))
xref = xref_next(xref);
}
return xref;
}
static void xref_jump(xref_t xref, struct el* elem)
{
int x, y;
if (!elem || !xref || xref_visible(xref, elem) ||
xref_rel_position(xref, elem, &x, &y))
return;
gfx_warp_cursor(elem->x + x, elem->y + y);
}
static int select_ref(int prev, int last)
{
int x, y;
struct el *elem = NULL;
xref_t xref = NULL;
gfx_cursor(&x, &y);
xref = look_xref(x, y, &elem);
if (!elem) {
if (!sel_el)
select_frame(0);
elem = el(sel_el);
}
if (last) {
if (!(xref = get_xref(elem->id, !prev)))
return -1;
} else if (xref) {
if (prev) {
if (!(xref = xref_prev(xref)) || xref_visible(xref, elem)) {
if (!box_isscroll_up(elem->id) || !box_isscroll_down(elem->id))
return -1;
else
xref = get_xref(elem->id, 1);
}
} else {
if (!(xref = xref_next(xref)) || xref_visible(xref, elem)) {
if (!box_isscroll_down(elem->id) || !box_isscroll_up(elem->id))
return -1;
else
xref = get_xref(elem->id, 0);
}
}
}
if (!xref)
xref = get_nearest_xref(elem->id, x, y);
if (!xref)
return -1;
xref_jump(xref, elem);
return 0;
}
static void game_scroll_up(int count)
{
int xm, ym;
struct el *o;
gfx_cursor(&xm, &ym);
o = look_obj(xm, ym);
if (o && (o->id == el_scene || o->id == el_inv)) {
scroll_up(o->id, count);
}
}
static void game_scroll_down(int count)
{
int xm, ym;
struct el *o;
gfx_cursor(&xm, &ym);
o = look_obj(xm, ym);
if (o && (o->id == el_scene || o->id == el_inv)) {
scroll_down(o->id, count);
}
}
static int game_scroll_pup(void)
{
int xm, ym;
struct el *o;
gfx_cursor(&xm, &ym);
o = look_obj(xm, ym);
if (o && (o->id == el_scene || o->id == el_inv)) {
return scroll_pup(o->id);
}
return -1;
}
static int game_scroll_pdown(void)
{
int xm, ym;
struct el *o;
gfx_cursor(&xm, &ym);
o = look_obj(xm, ym);
if (o && (o->id == el_scene || o->id == el_inv)) {
return scroll_pdown(o->id);
}
return -1;
}
static int is_key(struct inp_event *ev, const char *name)
{
return strcmp(ev->sym, name);
}
int game_pict_coord(int *x, int *y, int *w, int *h)
{
img_t img;
int ww, hh;
int xx, yy;
word_t word;
img = el_img(el_spic);
if (!img)
return -1;
if (game_theme.gfx_mode != GFX_MODE_EMBEDDED) {
xx = el(el_spic)->x;
yy = el(el_spic)->y;
ww = gfx_img_w(img);
hh = gfx_img_h(img);
goto out;
}
el_size(el_scene, &ww, &hh);
for (word = NULL; (word = txt_layout_words(txt_box_layout(el_box(el_scene)), word)); ) { /* scene */
if (word_image(word) != img) {
word = NULL;
/* first word is always pic */
break;
// continue;
}
word_geom(word, &xx, &yy, &ww, &hh);
yy -= txt_box_off(el_box(el_scene));
xx += el(el_scene)->x;
yy += el(el_scene)->y;
break;
}
if (!word)
return -1;
out:
if (x)
*x = xx;
if (y)
*y = yy;
if (w)
*w = ww;
if (h)
*h = hh;
return 0;
}
static int game_pic_click(int x, int y, int *ox, int *oy)
{
int xx, yy, ww, hh;
if (game_pict_coord(&xx, &yy, &ww, &hh))
return -1;
if (x >= xx && y >= yy && x < (xx + ww) && y < (yy + hh)) {
*ox = x - xx;
*oy = y - yy;
if (ww)
*ox = (int)((float)(*ox) * (float)game_pic_w / (float)ww);
else
*ox = 0;
if (hh)
*oy = (int)((float)(*oy) * (float)game_pic_h / (float)hh);
else
*oy = 0;
return 0;
}
return -1;
}
static int game_bg_click(int x, int y, int *ox, int *oy)
{
struct el *o;
struct game_theme *t = &game_theme;
if (x < t->xoff || y < t->yoff || x >= (t->w - t->xoff) || y >= (t->h - t->yoff))
return -1;
o = look_obj(x, y);
if (o && (o->id == el_sup || o->id == el_sdown ||
o->id == el_iup || o->id == el_idown ||
o->id == el_menu_button))
return -1; /* ask Odyssey for that ;) */
*ox = (int)((float)(x - t->xoff) / (float)t->scale);
*oy = (int)((float)(y - t->yoff) / (float)t->scale);
return 0;
}
static int game_input(int down, const char *key, int x, int y, int mb)
{
char *p;
struct instead_args args[8];
int rc = 0;
char tx[16];
char ty[16];
char tpx[16];
char tpy[16];
char tmb[16];
if (game_paused())
return -1;
if (mb == -1) {
const char *k;
args[0].val = "kbd"; args[0].type = INSTEAD_STR;
args[1].val = (down)?"true":"false"; args[1].type = INSTEAD_BOOL;
k = (key)?key:"unknown";
args[2].val = (char*)k; args[2].type = INSTEAD_STR;
args[3].val = NULL;
} else {
int px = -1;
int py = -1;
game_pic_click(x, y, &px, &py); /* got picture coord */
if (game_bg_click(x, y, &x, &y)) /* no click on bg */
return -1;
snprintf(tx, sizeof(tx), "%d", x);
snprintf(ty, sizeof(ty), "%d", y);
snprintf(tmb, sizeof(tmb), "%d", mb);
args[0].val = "mouse"; args[0].type = INSTEAD_STR;
args[1].val = (down)?"true":"false"; args[1].type = INSTEAD_BOOL;
args[2].val = tmb; args[2].type = INSTEAD_NUM;
args[3].val = tx; args[3].type = INSTEAD_NUM;
args[4].val = ty; args[4].type = INSTEAD_NUM;
args[5].val = NULL;
if (px != -1) {
snprintf(tpx, sizeof(tpx), "%d", px);
snprintf(tpy, sizeof(tpy), "%d", py);
args[5].val = tpx; args[5].type = INSTEAD_NUM;
args[6].val = tpy; args[6].type = INSTEAD_NUM;
args[7].val = NULL;
}
}
if (instead_function("stead.input", args)) {
instead_clear();
return -1;
}
p = instead_retval(0); instead_clear();
if (!p)
return -1;
mouse_reset(0);
if (opt_click && mb != -1)
snd_play(game_theme.click, -1, 0);
rc = game_cmd(p); free(p);
mouse_restore();
return (rc)?-1:0;
}
extern char zip_game_dirname[];
extern int unpack(const char *zipfilename, const char *dirname);
#ifdef _USE_BROWSE
int game_from_disk(void)
{
int i = 0;
char *g, *p, *b, *d;
char dir[PATH_MAX];
char base[PATH_MAX];
#ifndef MAEMO
#ifndef S60
if (opt_fs) {
int old_menu = (menu_shown) ? cur_menu: -1;
opt_fs ^= 1;
game_restart();
if (old_menu != -1)
game_menu(old_menu);
}
#endif
#endif
mouse_cursor(1);
game_cursor(CURSOR_OFF);
browse_dialog = 1;
g = p = open_file_dialog();
browse_dialog = 0;
game_cursor(CURSOR_ON);
mouse_cursor(0);
gfx_flip();
if (!p)
return -1;
strcpy(dir, p);
strcpy(base, p);
d = dir; b = base;
i = strlen(d);
if (i && d[i - 1] != '/') { /* file */
if (!idf_magic(d)) {
d = dirname(d);
strcpy(b, d);
}
}
d = dirname(d);
b = basename(b);
#ifdef _USE_UNPACK
p = games_sw ? games_sw:game_local_games_path(1);
fprintf(stderr,"Trying to install: %s\n", g);
if (!unpack(g, p)) {
if (!zip_game_dirname[0])
return -1;
if (games_replace(p, zip_game_dirname))
goto clean;
p = zip_game_dirname;
} else if (zip_game_dirname[0]) { /* error, needs to clean */
goto clean;
#else
if (0) {
#endif
} else if (games_replace(d, b)) {
return -1;
} else
p = b;
game_done(0);
if (game_init(p)) {
game_error(p);
}
return 0;
#ifdef _USE_UNPACK
clean:
p = getpath(p, zip_game_dirname);
fprintf(stderr, "Cleaning: '%s'...\n", p);
remove_dir(p);
free(p);
return -1;
#endif
}
#endif
int game_loop(void)
{
static int alt_pressed = 0;
static int shift_pressed = 0;
static int x = 0, y = 0;
struct inp_event ev;
memset(&ev, 0, sizeof(struct inp_event));
while (game_running) {
int rc;
ev.x = -1;
// game_cursor(CURSOR_CLEAR); /* release bg */
while ((rc = input(&ev, 1)) == AGAIN);
if (rc == -1) {/* close */
break;
} else if (curgame_dir && (ev.type == KEY_DOWN || ev.type == KEY_UP)
&& !game_input((ev.type == KEY_DOWN), ev.sym, -1, -1, -1)) {
; /* all is done in game_input */
} else if (curgame_dir && (ev.type == MOUSE_DOWN || ev.type == MOUSE_UP)
&& !game_input((ev.type == MOUSE_DOWN), "mouse", ev.x, ev.y, ev.code)) {
; /* all is done in game_input */
} else if (((ev.type == KEY_DOWN) || (ev.type == KEY_UP)) &&
(!is_key(&ev, "left alt") || !is_key(&ev, "right alt"))) {
alt_pressed = (ev.type == KEY_DOWN) ? 1:0;
} else if (((ev.type == KEY_DOWN) || (ev.type == KEY_UP)) &&
(!is_key(&ev,"left shift") || !is_key(&ev, "right shift"))) {
shift_pressed = (ev.type == KEY_DOWN) ? 1:0;
} else if (ev.type == KEY_DOWN) {
if (!is_key(&ev, "escape")
#if defined(S60) || defined(_WIN32_WCE)
|| !is_key(&ev, "space")
#endif
#if defined(_WIN32_WCE)
|| (ev.code >= 0xc0 && ev.code <= 0xcf) ||
!is_key(&ev, "f1") ||
!is_key(&ev, "f2") ||
!is_key(&ev, "f3") ||
!is_key(&ev, "f4") ||
!is_key(&ev, "f5")
#endif
#ifdef ANDROID
|| ev.code == 118
#endif
) {
if (use_xref)
disable_use();
else
menu_toggle();
} else if (!is_key(&ev, "f1")) {
if (!menu_shown)
menu_toggle();
/*
} else if (!is_key(&ev, "f6")) {
mouse_reset(1);
game_menu(menu_games);
*/
} else if (!is_key(&ev, "f2") && curgame_dir) {
mouse_reset(1);
game_menu(menu_save);
} else if (!is_key(&ev, "f3") && curgame_dir) {
mouse_reset(1);
game_menu(menu_load);
} else if (!is_key(&ev, "f8") && curgame_dir && !menu_shown) {
if (game_saves_enabled())
game_save(9);
} else if (!is_key(&ev, "f9") && curgame_dir && !menu_shown) {
if (game_saves_enabled()) {
if (!access(game_save_path(0, 9), R_OK)) {
if (!game_reset())
game_load(9);
}
}
} else if (!is_key(&ev, "f5") && curgame_dir && !menu_shown) {
mouse_reset(1);
game_cmd("look");
} else if (alt_pressed && !is_key(&ev, "r") && curgame_dir && !menu_shown && debug_sw) {
mouse_reset(1);
game_menu_act("/new");
shift_pressed = alt_pressed = 0;
} else if (!is_key(&ev, "f10")
#ifdef ANDROID
|| ev.code == 270
#endif
) {
#ifdef ANDROID
break;
#else
mouse_reset(1);
game_menu(menu_askquit);
#endif
} else if (alt_pressed && (!is_key(&ev, "q") || !is_key(&ev, "f4"))) {
break;
} else if (alt_pressed &&
(!is_key(&ev, "enter") || !is_key(&ev, "return"))) {
int old_menu = (menu_shown) ? cur_menu: -1;
shift_pressed = alt_pressed = 0;
opt_fs ^= 1;
game_restart();
if (old_menu != -1)
game_menu(old_menu);
} else if (!is_key(&ev, "f4") && !alt_pressed) {
#ifdef _USE_UNPACK
#ifdef _USE_BROWSE
mouse_reset(1);
if (!game_from_disk()) {
shift_pressed = alt_pressed = 0;
}
#endif
#endif
} else if (DIRECT_MODE && !menu_shown) {
; // nothing todo
} else if (!alt_pressed && (!is_key(&ev, "return") || !is_key(&ev, "enter")
#ifdef S60
|| !is_key(&ev, ".")
#endif
)) {
gfx_cursor(&x, &y);
game_highlight(-1, -1, 0); /* reset */
game_click(x, y, 0, 0);
game_highlight(x, y, 1); /* hl on/off */
game_highlight(x, y, 0);
if (game_click(x, y, 1, 0) == -1)
break;
} else if (!is_key(&ev, "tab")) {
select_frame(shift_pressed);
} else if (!is_key(&ev, "up") || !is_key(&ev, "down") ||
!is_key(&ev, "[8]") || !is_key(&ev, "[2]")) {
int lm;
int prev = !is_key(&ev, "up") || !is_key(&ev, "[8]");
if (opt_kbd == KBD_INVERSE)
lm = (alt_pressed || shift_pressed);
else
lm = (!alt_pressed && !shift_pressed);
if (menu_shown || lm) {
if (select_ref(prev, 0)) {
if (opt_kbd == KBD_SMART) {
(prev)?game_scroll_up(1):game_scroll_down(1);
select_ref(prev, 1);
} else
select_ref(prev, 1);
}
} else
(prev)?game_scroll_up(1):game_scroll_down(1);
} else if (!is_key(&ev, "page up") || !is_key(&ev, "[9]") ||
!is_key(&ev, "page down") || !is_key(&ev, "[3]")) {
int lm;
int prev = !is_key(&ev, "page up") || !is_key(&ev, "[9]");
if (opt_kbd == KBD_INVERSE)
lm = (alt_pressed || shift_pressed);
else
lm = (!alt_pressed && !shift_pressed);
if (menu_shown || lm) {
if (select_ref(prev, 0) || select_ref(prev, 1)) {
if (opt_kbd == KBD_SMART) {
int s = (prev)?game_scroll_pup():game_scroll_pdown();
if (!s)
select_ref(!prev, 1);
} else
select_ref(prev, 0);
}
} else {
if (prev)
game_scroll_pup();
else
game_scroll_pdown();
}
#if !defined(S60) && !defined(_WIN32_WCE)
} else if (!is_key(&ev, "left") || !is_key(&ev, "[4]")) {
select_ref(1, 0);
} else if (!is_key(&ev, "right") || !is_key(&ev, "[6]")) {
select_ref(0, 0);
} else if (!is_key(&ev, "backspace") && !menu_shown) {
scroll_pup(el_scene);
} else if (!is_key(&ev, "space") && !menu_shown) {
scroll_pdown(el_scene);
#else
} else if (!is_key(&ev, "left") || !is_key(&ev, "[4]")) {
if (menu_shown)
select_ref(1, 0);
else
select_frame(1);
} else if (!is_key(&ev, "right") || !is_key(&ev, "[6]")) {
if (menu_shown)
select_ref(0, 0);
else
select_frame(0);
#endif
}
} else if (DIRECT_MODE && !menu_shown) {
; // nothing todo
} else if (ev.type == MOUSE_DOWN) {
if (ev.code != 1)
disable_use();
else {
game_highlight(-1, -1, 0);
game_click(ev.x, ev.y, 0, 1);
x = ev.x;
y = ev.y;
}
} else if (ev.type == MOUSE_UP) {
game_highlight(-1, -1, 0);
if (game_click(ev.x, ev.y, 1, 1) == -1)
break;
} else if (ev.type == MOUSE_WHEEL_UP && !menu_shown) {
game_scroll_up(ev.count);
} else if (ev.type == MOUSE_WHEEL_DOWN && !menu_shown) {
game_scroll_down(ev.count);
} else if (ev.type == MOUSE_MOTION) {
if (motion_mode) {
motion_mode = 2;
scroll_motion(motion_id, motion_y - ev.y);
motion_y = ev.y;
}
// game_highlight(ev.x, ev.y, 1);
}
if (!DIRECT_MODE || menu_shown) {
if (old_xref)
game_highlight(x, y, 1);
else {
int x, y;
gfx_cursor(&x, &y);
game_highlight(x, y, 1);
}
}
game_cursor(CURSOR_ON);
if (err_msg) {
mouse_reset(1);
game_menu(menu_warning);
}
}
return 0;
}