mirror of
https://gitea.nishi.boats/pyrite-dev/milsko
synced 2026-01-06 17:39:45 +00:00
git-svn-id: http://svn2.nishi.boats/svn/milsko/trunk@468 b9cfdab3-6d41-4d17-bbe4-086880011989
215 lines
4.8 KiB
C
215 lines
4.8 KiB
C
/* $Id$ */
|
|
#include <Mw/Milsko.h>
|
|
|
|
#include "../../external/stb_ds.h"
|
|
#include "Mw/TypeDefs.h"
|
|
|
|
static void set_xywh(MwWidget handle) {
|
|
int height = 0;
|
|
int i;
|
|
MwMenu m = handle->internal;
|
|
|
|
for(i = 0; i < arrlen(m->sub); i++) {
|
|
int h = MwTextHeight(handle, m->sub[i]->name);
|
|
if(height < h) {
|
|
height = h;
|
|
}
|
|
}
|
|
|
|
height += 20;
|
|
|
|
MwVaApply(handle,
|
|
MwNx, 0,
|
|
MwNy, 0,
|
|
MwNwidth, MwGetInteger(handle->parent, MwNwidth),
|
|
MwNheight, height,
|
|
NULL);
|
|
}
|
|
|
|
static int create(MwWidget handle) {
|
|
MwMenu m = malloc(sizeof(*m));
|
|
|
|
m->name = NULL;
|
|
m->wsub = NULL;
|
|
m->sub = NULL;
|
|
handle->internal = m;
|
|
|
|
MwSetDefault(handle);
|
|
|
|
set_xywh(handle);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void recursive_free(MwMenu m) {
|
|
int i;
|
|
|
|
for(i = 0; i < arrlen(m->sub); i++) {
|
|
recursive_free(m->sub[i]);
|
|
}
|
|
|
|
if(m->sub != NULL) arrfree(m->sub);
|
|
if(m->name != NULL) free(m->name);
|
|
free(m);
|
|
}
|
|
|
|
static void destroy(MwWidget handle) {
|
|
MwMenu m = handle->internal;
|
|
|
|
recursive_free(m);
|
|
}
|
|
|
|
#define MENU_LOOP_DECL \
|
|
int i; \
|
|
MwMenu m = handle->internal; \
|
|
MwPoint p; \
|
|
MwRect r; \
|
|
\
|
|
p.x = 10; \
|
|
p.y = MwGetInteger(handle, MwNheight) / 2; \
|
|
\
|
|
r.x = 0; \
|
|
r.y = 0; \
|
|
r.width = MwGetInteger(handle, MwNwidth); \
|
|
r.height = MwGetInteger(handle, MwNheight);
|
|
|
|
#define BEGIN_MENU_LOOP \
|
|
for(i = 0; i < arrlen(m->sub); i++) { \
|
|
int incr = m->sub[i]->name[0] == '?' ? 1 : 0; \
|
|
int tw = MwTextWidth(handle, m->sub[i]->name + incr); \
|
|
int th = MwTextHeight(handle, m->sub[i]->name + incr); \
|
|
int oldx = p.x; \
|
|
int in_area; \
|
|
\
|
|
if(incr) { \
|
|
p.x = MwGetInteger(handle, MwNwidth) - tw - 10; \
|
|
} \
|
|
\
|
|
r.x = p.x - 5; \
|
|
r.y = p.y - th / 2 - 5; \
|
|
r.width = tw + 10; \
|
|
r.height = th + 10; \
|
|
\
|
|
in_area = (r.x <= handle->mouse_point.x && r.y <= handle->mouse_point.y && handle->mouse_point.x <= (int)(r.x + r.width) && handle->mouse_point.y <= (int)(r.y + r.height)) ? 1 : 0;
|
|
|
|
#define END_MENU_LOOP \
|
|
p.x += tw / 2 + 20; \
|
|
if(incr) p.x = oldx; \
|
|
}
|
|
|
|
static void draw(MwWidget handle) {
|
|
MwLLColor base = MwParseColor(handle, MwGetText(handle, MwNbackground));
|
|
MwLLColor text = MwParseColor(handle, MwGetText(handle, MwNforeground));
|
|
MENU_LOOP_DECL;
|
|
|
|
MwDrawWidgetBack(handle, &r, base, 0, MwTRUE);
|
|
|
|
BEGIN_MENU_LOOP;
|
|
if(m->sub[i]->wsub != NULL) {
|
|
MwDrawWidgetBack(handle, &r, base, 0, MwFALSE);
|
|
} else if(in_area && handle->pressed) {
|
|
MwDrawWidgetBack(handle, &r, base, 0, MwFALSE);
|
|
}
|
|
|
|
MwDrawText(handle, &p, m->sub[i]->name + incr, 1, MwALIGNMENT_BEGINNING, text);
|
|
END_MENU_LOOP;
|
|
|
|
MwLLFreeColor(text);
|
|
MwLLFreeColor(base);
|
|
}
|
|
|
|
static void parent_resize(MwWidget handle) {
|
|
set_xywh(handle);
|
|
}
|
|
|
|
static void mouse_down(MwWidget handle, void* ptr) {
|
|
MENU_LOOP_DECL;
|
|
|
|
if(((MwLLMouse*)ptr)->button != MwLLMouseLeft) return;
|
|
|
|
BEGIN_MENU_LOOP;
|
|
if(in_area) {
|
|
if(m->sub[i]->wsub == NULL && arrlen(m->sub[i]->sub) > 0) {
|
|
MwPoint p2;
|
|
|
|
p2.x = p.x - 5;
|
|
p2.y = p.y + th / 2 + 5;
|
|
|
|
m->sub[i]->wsub = MwCreateWidget(MwSubMenuClass, "submenu", handle, 0, 0, 0, 0);
|
|
MwSubMenuAppear(m->sub[i]->wsub, m->sub[i], &p2);
|
|
} else if(m->sub[i]->wsub != NULL && m->sub[i]->keep) {
|
|
MwDestroyWidget(m->sub[i]->wsub);
|
|
m->sub[i]->wsub = NULL;
|
|
m->sub[i]->keep = 0;
|
|
} else if(arrlen(m->sub[i]->sub) == 0) {
|
|
MwDispatchUserHandler(handle, MwNmenuHandler, m->sub[i]);
|
|
}
|
|
} else if(m->sub[i]->keep && m->sub[i]->wsub != NULL) {
|
|
MwDestroyWidget(m->sub[i]->wsub);
|
|
m->sub[i]->wsub = NULL;
|
|
m->sub[i]->keep = 0;
|
|
}
|
|
END_MENU_LOOP;
|
|
|
|
MwForceRender(handle);
|
|
}
|
|
|
|
static void mouse_up(MwWidget handle, void* ptr) {
|
|
MENU_LOOP_DECL;
|
|
|
|
if(((MwLLMouse*)ptr)->button != MwLLMouseLeft) return;
|
|
|
|
BEGIN_MENU_LOOP;
|
|
if(in_area && m->sub[i]->wsub != NULL) {
|
|
m->sub[i]->keep = 1;
|
|
} else if(m->sub[i]->wsub != NULL) {
|
|
MwDestroyWidget(m->sub[i]->wsub);
|
|
m->sub[i]->wsub = NULL;
|
|
m->sub[i]->keep = 0;
|
|
}
|
|
END_MENU_LOOP;
|
|
|
|
MwForceRender(handle);
|
|
}
|
|
|
|
static MwMenu mwMenuAddImpl(MwWidget handle, MwMenu menu, const char* name) {
|
|
MwMenu m = menu == NULL ? handle->internal : menu;
|
|
MwMenu new = malloc(sizeof(*new));
|
|
new->name = MwStringDupliacte(name);
|
|
new->sub = NULL;
|
|
new->wsub = NULL;
|
|
new->keep = 0;
|
|
|
|
arrput(m->sub, new);
|
|
|
|
set_xywh(handle);
|
|
|
|
return new;
|
|
}
|
|
|
|
static void func_handler(MwWidget handle, const char* name, void* out, va_list va) {
|
|
if(strcmp(name, "mwMenuAdd") == 0) {
|
|
MwMenu menu = va_arg(va, MwMenu);
|
|
const char* name = va_arg(va, const char*);
|
|
*(MwMenu*)out = mwMenuAddImpl(handle, menu, name);
|
|
}
|
|
}
|
|
|
|
MwClassRec MwMenuClassRec = {
|
|
create, /* create */
|
|
destroy, /* destroy */
|
|
draw, /* draw */
|
|
NULL, /* click */
|
|
parent_resize, /* parent_resize */
|
|
NULL, /* prop_change */
|
|
NULL, /* mouse_move */
|
|
mouse_up, /* mouse_up */
|
|
mouse_down, /* mouse_down */
|
|
NULL, /* key */
|
|
func_handler, /* custom */
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL};
|
|
MwClass MwMenuClass = &MwMenuClassRec;
|