/* $Id$ */ #include #include "../../external/stb_ds.h" static int get_first_entry(MwWidget handle, MwListBox lb) { int st = 0; int y = MwGetInteger(handle, MwNhasHeading) ? 1 : 0; st = MwGetInteger(lb->vscroll, MwNvalue); st = st * (MwGetInteger(lb->vscroll, MwNmaxValue) - MwGetInteger(lb->vscroll, MwNareaShown)) / MwGetInteger(lb->vscroll, MwNmaxValue); if(st < y) st = y; return st; } static int get_col_width(MwListBox lb, int ind) { int total = MwGetInteger(lb->frame, MwNwidth); int wid, i; if(arrlen(lb->width) <= ind) { wid = 0; } else { wid = lb->width[ind]; } if(wid > 0) return wid; if(wid == 0) { for(i = 0; i < arrlen(lb->width) && i < ind; i++) { int w = lb->width[i]; if(w < 0) { total -= total + w; } else if(wid > 0) { total -= w; } } return total; } if(wid < 0) { for(i = 0; i < arrlen(lb->width) && i < ind; i++) { int w = lb->width[i]; if(w < 0) { total -= total + w; } else if(wid > 0) { total -= w; } } return total + wid; } return 0; } static void vscroll_changed(MwWidget handle, void* user, void* call) { MwListBox lb = handle->parent->internal; (void)user; (void)call; MwForceRender(lb->frame); } static void frame_mouse_down(MwWidget handle, void* user, void* call) { MwListBox lb = handle->parent->internal; MwLLMouse* m = call; (void)user; if(m->button == MwLLMouseLeft) { int st = 0; int i; int y = MwDefaultBorderWidth; int h = MwGetInteger(handle, MwNheight); st = get_first_entry(handle->parent, lb); for(i = 0; (st + i) < arrlen(lb->list) && i < (h - MwDefaultBorderWidth * 2) / MwTextHeight(handle, "M"); i++) { if(y <= m->point.y && m->point.y <= (y + MwTextHeight(handle, "M"))) { unsigned long t; int old = lb->selected; lb->selected = st + i; if(((t = MwLLGetTick()) - lb->click_time) < 250 && old == st + i) { MwDispatchUserHandler(handle->parent, MwNactivateHandler, &lb->selected); } lb->click_time = t; } y += MwTextHeight(handle, "M"); } MwForceRender(lb->frame); lb->pressed = 1; } } static void frame_mouse_up(MwWidget handle, void* user, void* call) { MwListBox lb = handle->parent->internal; MwLLMouse* m = call; (void)user; if(m->button == MwLLMouseLeft) { lb->pressed = 0; } } static void frame_mouse_move(MwWidget handle, void* user, void* call) { MwListBox lb = handle->parent->internal; MwPoint* p = call; (void)user; if(lb->pressed) { int st = 0; int i; int y = MwDefaultBorderWidth; int h = MwGetInteger(handle, MwNheight); st = get_first_entry(handle->parent, lb); for(i = 0; (st + i) < arrlen(lb->list) && i < (h - MwDefaultBorderWidth * 2) / MwTextHeight(handle, "M"); i++) { if(y <= p->y && p->y <= (y + MwTextHeight(handle, "M"))) { lb->selected = st + i; } y += MwTextHeight(handle, "M"); } MwForceRender(lb->frame); } } static void frame_draw(MwWidget handle) { MwRect r; MwListBox lb = handle->parent->internal; MwLLColor base = MwParseColor(handle, MwGetText(handle, MwNbackground)); MwLLColor text = MwParseColor(handle, MwGetText(handle, MwNforeground)); int i; MwPoint p; int st = 0; r.x = 0; r.y = 0; r.width = MwGetInteger(handle, MwNwidth); r.height = MwGetInteger(handle, MwNheight); p.x = MwDefaultBorderWidth; p.y = MwDefaultBorderWidth; st = get_first_entry(handle->parent, lb); for(i = st; i < arrlen(lb->list) && i < st + (r.height - MwDefaultBorderWidth * 2) / MwTextHeight(handle, "M"); i++) { int selected = lb->selected == i ? 1 : 0; int j; if(selected) { MwRect r2; r2.x = 0; r2.y = p.y; r2.width = r.width; r2.height = MwTextHeight(handle, "M"); MwDrawRect(handle, &r2, text); } if(lb->list[i].pixmap != NULL) { MwRect r2; int h = (lb->list[i].pixmap->height > MwTextHeight(handle, "M")) ? MwTextHeight(handle, "M") : lb->list[i].pixmap->height; r2.x = MwDefaultBorderWidth; r2.y = p.y + (MwTextHeight(handle, "M") - h) / 2; r2.width = h * lb->list[i].pixmap->width / lb->list[i].pixmap->height; r2.height = h; MwLLDrawPixmap(handle->lowlevel, &r2, lb->list[i].pixmap); } p.y += MwTextHeight(handle, "M") / 2; p.x = MwGetInteger(handle->parent, MwNleftPadding); for(j = 0; j < arrlen(lb->list[i].name); j++) { p.x += MwDefaultBorderWidth; MwDrawText(handle, &p, lb->list[i].name[j], 0, MwALIGNMENT_BEGINNING, selected ? base : text); p.x += get_col_width(lb, j) - MwDefaultBorderWidth; if(j == 0) p.x -= MwGetInteger(handle->parent, MwNleftPadding); } p.y += MwTextHeight(handle, "M") / 2; } MwDrawFrame(handle, &r, base, 1); MwLLFreeColor(text); MwLLFreeColor(base); } static void resize(MwWidget handle) { MwListBox lb = handle->internal; int w = MwGetInteger(handle, MwNwidth); int h = MwGetInteger(handle, MwNheight); int ih, y; y = MwGetInteger(handle, MwNhasHeading) ? (MwTextHeight(handle, "M") + MwDefaultBorderWidth * 2) : 0; if(lb->vscroll == NULL) { lb->vscroll = MwVaCreateWidget(MwScrollBarClass, "vscroll", handle, w - 16, 0, 16, h, NULL); MwAddUserHandler(lb->vscroll, MwNchangedHandler, vscroll_changed, NULL); } else { MwVaApply(lb->vscroll, MwNx, w - 16, MwNy, 0, MwNwidth, 16, MwNheight, h, NULL); } h -= y; if(lb->frame == NULL) { lb->frame = MwVaCreateWidget(MwFrameClass, "frame", handle, 0, y, w - 16, h, NULL); lb->frame->draw_inject = frame_draw; MwAddUserHandler(lb->frame, MwNmouseDownHandler, frame_mouse_down, NULL); MwAddUserHandler(lb->frame, MwNmouseUpHandler, frame_mouse_up, NULL); MwAddUserHandler(lb->frame, MwNmouseMoveHandler, frame_mouse_move, NULL); } else { MwVaApply(lb->frame, MwNx, 0, MwNy, y, MwNwidth, w - 16, MwNheight, h, NULL); } h -= MwDefaultBorderWidth * 2; ih = arrlen(lb->list); if(ih == 0) ih = 1; MwVaApply(lb->vscroll, MwNareaShown, h / MwTextHeight(handle, "M"), MwNmaxValue, ih, NULL); } static int create(MwWidget handle) { MwListBox lb = malloc(sizeof(*lb)); memset(lb, 0, sizeof(*lb)); handle->internal = lb; MwSetDefault(handle); resize(handle); lb->list = NULL; lb->selected = -1; lb->click_time = 0; lb->width = NULL; MwSetInteger(handle, MwNleftPadding, 0); MwSetInteger(handle, MwNhasHeading, 0); return 0; } static void destroy(MwWidget handle) { MwListBox lb = handle->internal; MwListBoxReset(handle); arrfree(lb->list); arrfree(lb->width); free(handle->internal); } static void draw(MwWidget handle) { MwRect r; MwLLColor base = MwParseColor(handle, MwGetText(handle, MwNbackground)); MwLLColor text = MwParseColor(handle, MwGetText(handle, MwNforeground)); MwListBox lb = handle->internal; r.x = 0; r.y = 0; r.width = MwGetInteger(handle, MwNwidth); r.height = MwGetInteger(handle, MwNheight); MwDrawRect(handle, &r, base); if(MwGetInteger(handle, MwNhasHeading) && arrlen(lb->list) > 0) { MwPoint p; int i; int x = 0; r.width -= 16; for(i = 0; i < arrlen(lb->list[0].name); i++) { r.x = x; r.y = 0; r.width = get_col_width(lb, i); r.height = MwDefaultBorderWidth * 2 + MwTextHeight(handle, "M"); MwDrawFrame(handle, &r, base, 0); x += MwDefaultBorderWidth; p.x = x; p.y = r.y + r.height / 2; MwDrawText(handle, &p, lb->list[0].name[i], 0, MwALIGNMENT_BEGINNING, text); x += r.width + MwDefaultBorderWidth; } } MwLLFreeColor(text); MwLLFreeColor(base); } static void prop_change(MwWidget handle, const char* prop) { if(strcmp(prop, MwNwidth) == 0 || strcmp(prop, MwNheight) == 0 || strcmp(prop, MwNhasHeading) == 0) resize(handle); if(strcmp(prop, MwNleftPadding) == 0) { MwListBox lb = handle->internal; MwForceRender(lb->frame); } if(strcmp(prop, MwNhasHeading) == 0) { MwForceRender(handle); } } void MwListBoxVaInsert(MwWidget handle, int index, MwLLPixmap pixmap, va_list va) { MwListBox lb = handle->internal; MwListBoxEntry entry; char* name; entry.name = NULL; while((name = va_arg(va, char*)) != NULL) { name = MwStringDupliacte(name); arrput(entry.name, name); } entry.pixmap = pixmap; if(index == -1) index = arrlen(lb->list); arrins(lb->list, index, entry); resize(handle); if(index < (MwGetInteger(lb->vscroll, MwNvalue) + MwGetInteger(lb->vscroll, MwNareaShown))) { MwForceRender(lb->frame); } } void MwListBoxVaInsertMultiple(MwWidget handle, int index, int count, MwLLPixmap* pixmap, va_list va) { int i; MwListBox lb = handle->internal; int old; char*** vlist = NULL; char** list; if(index == -1) index = arrlen(lb->list); old = index; while((list = va_arg(va, char**)) != NULL) { arrput(vlist, list); } for(i = 0; i < count; i++) { MwListBoxEntry entry; char* name; int j; entry.name = NULL; for(j = 0; j < arrlen(vlist); j++) { if(vlist[j][i] == NULL) continue; name = MwStringDupliacte(vlist[j][i]); arrput(entry.name, name); } entry.pixmap = NULL; if(pixmap != NULL) entry.pixmap = pixmap[i]; arrins(lb->list, index, entry); index++; } arrfree(vlist); resize(handle); if(old < (MwGetInteger(lb->vscroll, MwNvalue) + MwGetInteger(lb->vscroll, MwNareaShown))) { MwForceRender(lb->frame); } } static void mwListBoxDeleteImpl(MwWidget handle, int index) { MwListBox lb = handle->internal; int i; if(index == -1) index = arrlen(lb->list) - 1; for(i = 0; i < arrlen(lb->list[index].name); i++) { free(lb->list[index].name[i]); } arrfree(lb->list[index].name); arrdel(lb->list, index); if(lb->selected >= arrlen(lb->list)) { lb->selected = arrlen(lb->list) - 1; } if(lb->selected < 0) { lb->selected = -1; } resize(handle); if(index < (MwGetInteger(lb->vscroll, MwNvalue) + MwGetInteger(lb->vscroll, MwNareaShown))) { MwForceRender(lb->frame); } } static void mwListBoxResetImpl(MwWidget handle) { MwListBox lb = handle->internal; int i; while(arrlen(lb->list) > 0) { for(i = 0; i < arrlen(lb->list[0].name); i++) { if(lb->list[0].name[i] != NULL) free(lb->list[0].name[i]); } arrfree(lb->list[0].name); arrdel(lb->list, 0); } lb->selected = -1; resize(handle); MwForceRender(lb->frame); } void MwListBoxInsertMultiple(MwWidget handle, int index, int count, MwLLPixmap* pixmap, ...) { va_list va; va_start(va, pixmap); MwListBoxVaInsertMultiple(handle, index, count, pixmap, va); va_end(va); } void MwListBoxInsert(MwWidget handle, int index, MwLLPixmap pixmap, ...) { va_list va; va_start(va, pixmap); MwListBoxVaInsert(handle, index, pixmap, va); va_end(va); } static const char* mwListBoxGetImpl(MwWidget handle, int index) { MwListBox lb = handle->internal; if(index < 0) return NULL; if(index >= arrlen(lb->list)) return NULL; return lb->list[index].name[0]; } static void mwListBoxSetWidthImpl(MwWidget handle, int index, int width) { MwListBox lb = handle->internal; while(((index + 1) - arrlen(lb->width)) > 0) arrput(lb->width, 0); lb->width[index] = width; MwForceRender(handle); MwForceRender(lb->frame); } static void func_handler(MwWidget handle, const char* name, void* out, va_list va) { (void)out; if(strcmp(name, "mwListBoxDelete") == 0) { int index = va_arg(va, int); mwListBoxDeleteImpl(handle, index); } if(strcmp(name, "mwListBoxReset") == 0) { mwListBoxResetImpl(handle); } if(strcmp(name, "mwListBoxGet") == 0) { int index = va_arg(va, int); mwListBoxGetImpl(handle, index); } if(strcmp(name, "mwListBoxSetWidth") == 0) { int index = va_arg(va, int); int width = va_arg(va, int); mwListBoxSetWidthImpl(handle, index, width); } } MwClassRec MwListBoxClassRec = { create, /* create */ destroy, /* destroy */ draw, /* draw */ NULL, /* click */ NULL, /* parent_resize */ prop_change, /* prop_change */ NULL, /* mouse_move */ NULL, /* mouse_up */ NULL, /* mouse_down */ NULL, /* key */ func_handler, /* custom */ NULL, NULL, NULL, NULL}; MwClass MwListBoxClass = &MwListBoxClassRec;