mirror of
https://gitea.nishi.boats/pyrite-dev/milsko
synced 2025-12-30 22:20:50 +00:00
262 lines
6.3 KiB
C
262 lines
6.3 KiB
C
#include <Mw/Milsko.h>
|
|
|
|
#include "../../external/stb_ds.h"
|
|
|
|
static int create(MwWidget handle) {
|
|
MwLLBeginStateChange(handle->lowlevel);
|
|
|
|
MwSetDefault(handle);
|
|
MwSetInteger(handle, MwNleftPadding, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void null_all(MwMenu menu) {
|
|
int i;
|
|
for(i = 0; i < arrlen(menu->sub); i++) {
|
|
null_all(menu->sub[i]);
|
|
}
|
|
menu->wsub = NULL;
|
|
}
|
|
|
|
static void destroy(MwWidget handle) {
|
|
MwMenu menu = handle->internal;
|
|
|
|
menu->wsub = NULL;
|
|
null_all(menu);
|
|
}
|
|
|
|
static void draw(MwWidget handle) {
|
|
MwLLColor base = MwParseColor(handle, MwGetText(handle, MwNbackground));
|
|
MwLLColor text = MwParseColor(handle, MwGetText(handle, MwNforeground));
|
|
MwRect r;
|
|
MwMenu menu = handle->internal;
|
|
|
|
r.x = 0;
|
|
r.y = 0;
|
|
r.width = MwGetInteger(handle, MwNwidth);
|
|
r.height = MwGetInteger(handle, MwNheight);
|
|
|
|
MwDrawWidgetBack(handle, &r, base, 0, MwTRUE);
|
|
|
|
if(menu != NULL) {
|
|
MwPoint p;
|
|
int i;
|
|
|
|
p.x = 0;
|
|
p.y = 3;
|
|
|
|
for(i = 0; i < arrlen(menu->sub); i++) {
|
|
if(strcmp(menu->sub[i]->name, "----") == 0) {
|
|
MwRect rc;
|
|
|
|
p.y += 1;
|
|
|
|
rc.x = MwDefaultBorderWidth(handle) * 2;
|
|
rc.y = p.y;
|
|
rc.width = r.width - (rc.x * 2) - MwGetInteger(handle, MwNleftPadding);
|
|
rc.height = 2;
|
|
|
|
rc.x += MwGetInteger(handle, MwNleftPadding);
|
|
|
|
MwDrawFrameEx(handle, &rc, base, 1, 1, 0, 0);
|
|
|
|
p.y += 2 + 1;
|
|
} else {
|
|
int tw = MwTextWidth(handle, menu->sub[i]->name);
|
|
int th = MwTextHeight(handle, menu->sub[i]->name);
|
|
|
|
if(menu->sub[i]->wsub != NULL) {
|
|
r.x = MwGetInteger(handle, MwNleftPadding);
|
|
r.y = p.y - 3;
|
|
r.width = MwGetInteger(handle, MwNwidth) - MwGetInteger(handle, MwNleftPadding);
|
|
r.height = th + 3 * 2;
|
|
MwDrawWidgetBack(handle, &r, base, 0, MwTRUE);
|
|
}
|
|
|
|
p.x = 5 + tw / 2 + MwGetInteger(handle, MwNleftPadding);
|
|
|
|
p.y += th / 2;
|
|
MwDrawText(handle, &p, menu->sub[i]->name, menu->sub[i]->wsub != NULL ? 1 : 0, MwALIGNMENT_CENTER, text);
|
|
|
|
if(arrlen(menu->sub[i]->sub) > 0) {
|
|
MwRect tr;
|
|
|
|
tr.x = MwGetInteger(handle, MwNwidth) - MwDefaultBorderWidth(handle) - 11 - 2;
|
|
tr.y = p.y - th / 2 + 2;
|
|
tr.width = tr.height = 11;
|
|
|
|
MwDrawTriangle(handle, &tr, base, menu->sub[i]->wsub != NULL ? 1 : 0, MwEAST);
|
|
}
|
|
|
|
p.y += th / 2 + 3;
|
|
}
|
|
}
|
|
}
|
|
|
|
MwLLFreeColor(text);
|
|
MwLLFreeColor(base);
|
|
}
|
|
|
|
static void click(MwWidget handle) {
|
|
MwWidget w = handle;
|
|
jmp_buf jmp;
|
|
MwMenu menu = handle->internal;
|
|
|
|
if(arrlen(menu->sub) > 0) {
|
|
int ww = 0, i;
|
|
MwRect rc;
|
|
|
|
rc.x = 5;
|
|
rc.y = 3;
|
|
for(i = 0; i < arrlen(menu->sub); i++) {
|
|
int tw = MwTextWidth(handle, menu->sub[i]->name);
|
|
if(tw > ww) ww = tw;
|
|
}
|
|
|
|
rc.width = ww + 15;
|
|
for(i = 0; i < arrlen(menu->sub); i++) {
|
|
int th = MwTextHeight(handle, menu->sub[i]->name);
|
|
rc.height = th;
|
|
|
|
if(strcmp(menu->sub[i]->name, "----") == 0) {
|
|
rc.height = 2 - 1;
|
|
}
|
|
|
|
if(MwGetInteger(handle, MwNleftPadding) <= handle->mouse_point.x && rc.y <= handle->mouse_point.y && handle->mouse_point.y <= (int)(rc.y + rc.height)) {
|
|
if(menu->sub[i]->wsub == NULL && arrlen(menu->sub[i]->sub) > 0) {
|
|
MwPoint p;
|
|
int j;
|
|
|
|
for(j = 0; j < arrlen(menu->sub); j++) {
|
|
if(menu->sub[j]->wsub != NULL) MwDestroyWidget(menu->sub[j]->wsub);
|
|
menu->sub[j]->wsub = NULL;
|
|
}
|
|
|
|
p.x = MwGetInteger(handle, MwNwidth);
|
|
p.y = rc.y - 3;
|
|
|
|
menu->sub[i]->wsub = MwCreateWidget(MwSubMenuClass, "submenu", handle, 0, 0, 0, 0);
|
|
MwSubMenuAppear(menu->sub[i]->wsub, menu->sub[i], &p, 0);
|
|
i = -1;
|
|
} else if(menu->sub[i]->wsub != NULL && arrlen(menu->sub[i]->sub) > 0) {
|
|
while(w->parent->widget_class == MwSubMenuClass) w = w->parent;
|
|
|
|
MwDestroyWidget(menu->sub[i]->wsub);
|
|
menu->sub[i]->wsub = NULL;
|
|
|
|
MwForceRender(w->parent);
|
|
|
|
MwForceRender(handle);
|
|
} else if(strcmp(menu->sub[i]->name, "----") != 0 && arrlen(menu->sub[i]->sub) == 0) {
|
|
while(w->parent->widget_class == MwSubMenuClass) w = w->parent;
|
|
MwGetBeforeStep(w, &jmp);
|
|
|
|
MwDestroyWidget(w);
|
|
((MwMenu)w->internal)->wsub = NULL;
|
|
|
|
MwForceRender(w->parent);
|
|
|
|
MwDispatchUserHandler(w->parent, MwNmenuHandler, menu->sub[i]);
|
|
}
|
|
}
|
|
|
|
rc.y += rc.height + 3;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void mwSubMenuAppearImpl(MwWidget handle, MwMenu menu, MwPoint* point, int diff_calc) {
|
|
MwRect rc, sz;
|
|
MwPoint p = *point;
|
|
|
|
MwSubMenuGetSize(handle, menu, &sz);
|
|
|
|
MwGetScreenSize(handle, &rc);
|
|
|
|
handle->internal = menu;
|
|
|
|
if(diff_calc) {
|
|
p.y = p.y - sz.height;
|
|
}
|
|
|
|
MwLLMakeToolWindow(handle->lowlevel);
|
|
MwLLDetach(handle->lowlevel, &p);
|
|
|
|
if(MwGetInteger(handle, MwNy) + sz.height > rc.height) {
|
|
MwVaApply(handle,
|
|
MwNy, rc.height - sz.height,
|
|
NULL);
|
|
}
|
|
MwLLEndStateChange(handle->lowlevel);
|
|
|
|
MwVaApply(handle,
|
|
MwNwidth, sz.width,
|
|
MwNheight, sz.height,
|
|
NULL);
|
|
}
|
|
|
|
static void mwSubMenuGetSizeImpl(MwWidget handle, MwMenu menu, MwRect* rect) {
|
|
int i;
|
|
|
|
rect->width = 0;
|
|
rect->height = 0;
|
|
|
|
for(i = 0; i < arrlen(menu->sub); i++) {
|
|
if(strcmp(menu->sub[i]->name, "----") == 0) {
|
|
rect->height += 2 + 2;
|
|
} else {
|
|
int tw = MwTextWidth(handle, menu->sub[i]->name);
|
|
rect->height += MwTextHeight(handle, menu->sub[i]->name) + 3;
|
|
if(tw > rect->width) {
|
|
rect->width = tw;
|
|
}
|
|
}
|
|
}
|
|
|
|
rect->width += MwGetInteger(handle, MwNleftPadding);
|
|
|
|
rect->width += 10 + 15;
|
|
rect->height += 3;
|
|
|
|
rect->width += 16;
|
|
}
|
|
|
|
static void func_handler(MwWidget handle, const char* name, void* out, va_list va) {
|
|
(void)out;
|
|
|
|
if(strcmp(name, "mwSubMenuAppear") == 0) {
|
|
MwMenu menu = va_arg(va, MwMenu);
|
|
MwPoint* point = va_arg(va, MwPoint*);
|
|
int diff_calc = va_arg(va, int);
|
|
mwSubMenuAppearImpl(handle, menu, point, diff_calc);
|
|
} else if(strcmp(name, "mwSubMenuGetSize") == 0) {
|
|
MwMenu menu = va_arg(va, MwMenu);
|
|
MwRect* rect = va_arg(va, MwRect*);
|
|
mwSubMenuGetSizeImpl(handle, menu, rect);
|
|
}
|
|
}
|
|
|
|
MwClassRec MwSubMenuClassRec = {
|
|
create, /* create */
|
|
destroy, /* destroy */
|
|
draw, /* draw */
|
|
click, /* click */
|
|
NULL, /* parent_resize */
|
|
NULL, /* prop_change */
|
|
NULL, /* mouse_move */
|
|
MwForceRender2, /* mouse_up */
|
|
MwForceRender2, /* mouse_down */
|
|
NULL, /* key */
|
|
func_handler, /* execute */
|
|
NULL, /* tick */
|
|
NULL, /* resize */
|
|
NULL, /* children_update */
|
|
NULL, /* children_prop_change */
|
|
NULL, /* clipboard_received */
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL};
|
|
MwClass MwSubMenuClass = &MwSubMenuClassRec;
|