ltk

GUI toolkit for X11 (WIP)
git clone git://lumidify.org/ltk.git (fast, but not encrypted)
git clone https://lumidify.org/ltk.git (encrypted, but very slow)
git clone git://4kcetb7mo7hj6grozzybxtotsub5bempzo4lirzc3437amof2c2impyd.onion/ltk.git (over tor)
Log | Files | Refs | README | LICENSE

commit fcbc1ce6f1df72e293da709c8f5a29e30a1ac6ba
parent a69310e2abfdd32a215f15785b9c12cb47cbc32c
Author: lumidify <nobody@lumidify.org>
Date:   Fri,  3 May 2024 19:35:52 +0200

Add basic checkbutton

Diffstat:
MMakefile | 2++
Mexamples/ltk/test.c | 6++++++
Msrc/ltk/button.c | 2+-
Msrc/ltk/button.h | 1-
Asrc/ltk/checkbutton.c | 270+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Asrc/ltk/checkbutton.h | 37+++++++++++++++++++++++++++++++++++++
Msrc/ltk/config.c | 1+
Msrc/ltk/event_xlib.c | 44++++++++++++++++++++++----------------------
Msrc/ltk/widget.h | 2++
Msrc/ltk/widget_internal.h | 1+
10 files changed, 342 insertions(+), 24 deletions(-)

diff --git a/Makefile b/Makefile @@ -57,6 +57,7 @@ OBJ_LTK = \ src/ltk/widget.o \ src/ltk/ltk.o \ src/ltk/button.o \ + src/ltk/checkbutton.o \ src/ltk/graphics_xlib.o \ src/ltk/surface_cache.o \ src/ltk/event_xlib.o \ @@ -94,6 +95,7 @@ OBJ_TEST = examples/ltk/test.o # currently so short that I don't really care. HDR_LTK = \ src/ltk/button.h \ + src/ltk/checkbutton.h \ src/ltk/color.h \ src/ltk/label.h \ src/ltk/rect.h \ diff --git a/examples/ltk/test.c b/examples/ltk/test.c @@ -9,6 +9,7 @@ #include <ltk/entry.h> #include <ltk/menu.h> #include <ltk/box.h> +#include <ltk/checkbutton.h> int quit(ltk_widget *self, ltk_callback_arglist args, ltk_callback_arg data) { @@ -81,6 +82,11 @@ main(int argc, char *argv[]) { ltk_box_add(box, LTK_CAST_WIDGET(btn4), LTK_STICKY_LEFT); ltk_box_add(box, LTK_CAST_WIDGET(btn5), LTK_STICKY_LEFT); + ltk_checkbutton *cbtn1 = ltk_checkbutton_create(window, "Checkbutton1", 0); + ltk_checkbutton *cbtn2 = ltk_checkbutton_create(window, "Checkbutton2", 1); + ltk_box_add(box, LTK_CAST_WIDGET(cbtn1), LTK_STICKY_LEFT); + ltk_box_add(box, LTK_CAST_WIDGET(cbtn2), LTK_STICKY_LEFT); + ltk_grid_add(grid, LTK_CAST_WIDGET(menu), 0, 0, 1, 2, LTK_STICKY_LEFT|LTK_STICKY_RIGHT); ltk_grid_add(grid, LTK_CAST_WIDGET(button), 1, 0, 1, 1, LTK_STICKY_LEFT); ltk_grid_add(grid, LTK_CAST_WIDGET(button1), 1, 1, 1, 1, LTK_STICKY_RIGHT); diff --git a/src/ltk/button.c b/src/ltk/button.c @@ -14,9 +14,9 @@ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ -#include <stdint.h> #include <stdio.h> +#include "config.h" #include "button.h" #include "color.h" #include "graphics.h" diff --git a/src/ltk/button.h b/src/ltk/button.h @@ -17,7 +17,6 @@ #ifndef LTK_BUTTON_H #define LTK_BUTTON_H -#include "graphics.h" #include "text.h" #include "widget.h" #include "window.h" diff --git a/src/ltk/checkbutton.c b/src/ltk/checkbutton.c @@ -0,0 +1,270 @@ +/* + * Copyright (c) 2024 lumidify <nobody@lumidify.org> + * + * Permission to use, copy, modify, and/or distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +#include <stdio.h> + +#include "config.h" +#include "checkbutton.h" +#include "color.h" +#include "graphics.h" +#include "ltk.h" +#include "memory.h" +#include "rect.h" +#include "text.h" +#include "util.h" +#include "widget.h" + +#define MAX_CHECKBUTTON_BORDER_WIDTH 10000 +#define MAX_CHECKBUTTON_PADDING 50000 +#define MAX_CHECKBUTTON_BOX_SIZE 50000 + +static void ltk_checkbutton_draw(ltk_widget *self, ltk_surface *draw_surf, int x, int y, ltk_rect clip); +static int ltk_checkbutton_release(ltk_widget *self); +static void ltk_checkbutton_destroy(ltk_widget *self, int shallow); +static void ltk_checkbutton_recalc_ideal_size(ltk_widget *self); + +static struct ltk_widget_vtable vtable = { + .key_press = NULL, + .key_release = NULL, + .mouse_press = NULL, + .mouse_release = NULL, + .release = &ltk_checkbutton_release, + .motion_notify = NULL, + .mouse_leave = NULL, + .mouse_enter = NULL, + .change_state = NULL, + .get_child_at_pos = NULL, + .resize = NULL, + .hide = NULL, + .draw = &ltk_checkbutton_draw, + .destroy = &ltk_checkbutton_destroy, + .child_size_change = NULL, + .remove_child = NULL, + .recalc_ideal_size = &ltk_checkbutton_recalc_ideal_size, + .type = LTK_WIDGET_CHECKBUTTON, + .flags = LTK_NEEDS_REDRAW | LTK_ACTIVATABLE_ALWAYS, + .invalid_signal = LTK_CHECKBUTTON_SIGNAL_INVALID, +}; + +static struct { + ltk_color *text_color; + + ltk_color *fill; + ltk_color *fill_pressed; + ltk_color *fill_hover; + ltk_color *fill_active; + ltk_color *fill_disabled; + + ltk_color *box_fill; + ltk_color *box_border; + + ltk_color *box_fill_pressed; + ltk_color *box_border_pressed; + + ltk_color *box_fill_hover; + ltk_color *box_border_hover; + + ltk_color *box_fill_active; + ltk_color *box_border_active; + + ltk_color *box_fill_disabled; + ltk_color *box_border_disabled; + + ltk_color *box_fill_checked; + ltk_color *box_border_checked; + + ltk_color *box_fill_pressed_checked; + ltk_color *box_border_pressed_checked; + + ltk_color *box_fill_hover_checked; + ltk_color *box_border_hover_checked; + + ltk_color *box_fill_active_checked; + ltk_color *box_border_active_checked; + + ltk_color *box_fill_disabled_checked; + ltk_color *box_border_disabled_checked; + + char *font; + ltk_size box_size; + ltk_size box_border_width; + ltk_size pad; + ltk_size font_size; +} theme; + +static ltk_theme_parseinfo parseinfo[] = { + {"fill", THEME_COLOR, {.color = &theme.fill}, {.color = "#000000"}, 0, 0, 0}, + {"fill-hover", THEME_COLOR, {.color = &theme.fill_hover}, {.color = "#222222"}, 0, 0, 0}, + {"fill-active", THEME_COLOR, {.color = &theme.fill_active}, {.color = "#222222"}, 0, 0, 0}, + {"fill-disabled", THEME_COLOR, {.color = &theme.fill_disabled}, {.color = "#292929"}, 0, 0, 0}, + {"fill-pressed", THEME_COLOR, {.color = &theme.fill_pressed}, {.color = "#222222"}, 0, 0, 0}, + + {"box-fill", THEME_COLOR, {.color = &theme.box_fill}, {.color = "#000000"}, 0, 0, 0}, + {"box-fill-hover", THEME_COLOR, {.color = &theme.box_fill_hover}, {.color = "#222222"}, 0, 0, 0}, + {"box-fill-active", THEME_COLOR, {.color = &theme.box_fill_active}, {.color = "#222222"}, 0, 0, 0}, + {"box-fill-disabled", THEME_COLOR, {.color = &theme.box_fill_disabled}, {.color = "#292929"}, 0, 0, 0}, + {"box-fill-pressed", THEME_COLOR, {.color = &theme.box_fill_pressed}, {.color = "#222222"}, 0, 0, 0}, + {"box-border", THEME_COLOR, {.color = &theme.box_border}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"box-border-hover", THEME_COLOR, {.color = &theme.box_border_hover}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"box-border-active", THEME_COLOR, {.color = &theme.box_border_active}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"box-border-disabled", THEME_COLOR, {.color = &theme.box_border_disabled}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"box-border-pressed", THEME_COLOR, {.color = &theme.box_border_pressed}, {.color = "#FFFFFF"}, 0, 0, 0}, + + {"box-fill-checked", THEME_COLOR, {.color = &theme.box_fill_checked}, {.color = "#113355"}, 0, 0, 0}, + {"box-fill-hover-checked", THEME_COLOR, {.color = &theme.box_fill_hover_checked}, {.color = "#738194"}, 0, 0, 0}, + {"box-fill-active-checked", THEME_COLOR, {.color = &theme.box_fill_active_checked}, {.color = "#113355"}, 0, 0, 0}, + {"box-fill-disabled-checked", THEME_COLOR, {.color = &theme.box_fill_disabled_checked}, {.color = "#292929"}, 0, 0, 0}, + {"box-fill-pressed-checked", THEME_COLOR, {.color = &theme.box_fill_pressed_checked}, {.color = "#113355"}, 0, 0, 0}, + {"box-border-checked", THEME_COLOR, {.color = &theme.box_border_checked}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"box-border-hover-checked", THEME_COLOR, {.color = &theme.box_border_hover_checked}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"box-border-active-checked", THEME_COLOR, {.color = &theme.box_border_active_checked}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"box-border-disabled-checked", THEME_COLOR, {.color = &theme.box_border_disabled_checked}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"box-border-pressed-checked", THEME_COLOR, {.color = &theme.box_border_pressed_checked}, {.color = "#FFFFFF"}, 0, 0, 0}, + + {"box-size", THEME_SIZE, {.size = &theme.box_size}, {.size = {.val = 500, .unit = LTK_UNIT_MM}}, 0, MAX_CHECKBUTTON_BOX_SIZE, 0}, + {"box-border-width", THEME_SIZE, {.size = &theme.box_border_width}, {.size = {.val = 25, .unit = LTK_UNIT_MM}}, 0, MAX_CHECKBUTTON_BORDER_WIDTH, 0}, + {"pad", THEME_SIZE, {.size = &theme.pad}, {.size = {.val = 100, .unit = LTK_UNIT_MM}}, 0, MAX_CHECKBUTTON_PADDING, 0}, + {"text-color", THEME_COLOR, {.color = &theme.text_color}, {.color = "#FFFFFF"}, 0, 0, 0}, + {"font", THEME_STRING, {.str = &theme.font}, {.str = "Monospace"}, 0, 0, 0}, + {"font-size", THEME_SIZE, {.size = &theme.font_size}, {.size = {.val = 1200, .unit = LTK_UNIT_PT}}, 0, 20000, 0}, +}; + +void +ltk_checkbutton_get_theme_parseinfo(ltk_theme_parseinfo **p, size_t *len) { + *p = parseinfo; + *len = LENGTH(parseinfo); +} + +/* FIXME: a lot more theme settings */ +static void +ltk_checkbutton_draw(ltk_widget *self, ltk_surface *draw_surf, int x, int y, ltk_rect clip) { + ltk_checkbutton *button = LTK_CAST_CHECKBUTTON(self); + ltk_rect lrect = self->lrect; + ltk_rect clip_final = ltk_rect_intersect(clip, (ltk_rect){0, 0, lrect.w, lrect.h}); + if (clip_final.w <= 0 || clip_final.h <= 0) + return; + + int box_size = ltk_size_to_pixel(theme.box_size, self->last_dpi); + int box_bw = ltk_size_to_pixel(theme.box_border_width, self->last_dpi); + int pad = ltk_size_to_pixel(theme.pad, self->last_dpi); + ltk_color *fill = NULL, *box_border = NULL, *box_fill = NULL; + if (self->state & LTK_DISABLED) { + fill = theme.fill_disabled; + box_border = button->checked ? theme.box_border_disabled_checked : theme.box_border_disabled; + box_fill = button->checked ? theme.box_fill_disabled_checked : theme.box_fill_disabled; + } else if (self->state & LTK_PRESSED) { + fill = theme.fill_pressed; + box_border = button->checked ? theme.box_border_pressed_checked : theme.box_border_pressed; + box_fill = button->checked ? theme.box_fill_pressed_checked : theme.box_fill_pressed; + } else if (self->state & LTK_HOVER) { + fill = theme.fill_hover; + box_border = button->checked ? theme.box_border_hover_checked : theme.box_border_hover; + box_fill = button->checked ? theme.box_fill_hover_checked : theme.box_fill_hover; + } else if (self->state & LTK_ACTIVE) { + fill = theme.fill_active; + box_border = button->checked ? theme.box_border_active_checked : theme.box_border_active; + box_fill = button->checked ? theme.box_fill_active_checked : theme.box_fill_active; + } else { + fill = theme.fill; + box_border = button->checked ? theme.box_border_checked : theme.box_border; + box_fill = button->checked ? theme.box_fill_checked : theme.box_fill; + } + ltk_rect box_rect = {x + pad, y + pad, box_size, box_size}; + ltk_rect draw_clip = {x + clip_final.x, y + clip_final.y, clip_final.w, clip_final.h}; + ltk_rect box_clip = ltk_rect_intersect(box_rect, draw_clip); + ltk_surface_fill_rect(draw_surf, fill, draw_clip); + ltk_surface_fill_rect(draw_surf, box_fill, box_clip); + if (box_bw > 0) { + ltk_surface_draw_border_clipped( + draw_surf, box_border, box_rect, box_clip, box_bw, LTK_BORDER_ALL + ); + } + int text_w, text_h; + ltk_text_line_get_size(button->tl, &text_w, &text_h); + int text_x = x + 2 * pad + box_size; + int text_y = y + (lrect.h - text_h) / 2; + ltk_text_line_draw_clipped(button->tl, draw_surf, theme.text_color, text_x, text_y, draw_clip); + /* FIXME: only redraw if dirty (needs to be handled higher-up to only + call draw when dirty or window rect invalidated */ + self->dirty = 0; +} + +static int +ltk_checkbutton_release(ltk_widget *self) { + ltk_checkbutton *button = LTK_CAST_CHECKBUTTON(self); + button->checked = !button->checked; + ltk_widget_emit_signal(self, LTK_CHECKBUTTON_SIGNAL_CHANGED, LTK_EMPTY_ARGLIST); + return 1; +} + +int +ltk_checkbutton_get_checked(ltk_checkbutton *button) { + return button->checked; +} + +void +ltk_checkbutton_set_checked(ltk_checkbutton *button, int checked) { + button->checked = checked; + ltk_widget *self = LTK_CAST_WIDGET(button); + ltk_window_invalidate_widget_rect(self->window, self); +} + +#define MAX(a, b) ((a) > (b) ? (a) : (b)) + +static void +recalc_ideal_size(ltk_checkbutton *button) { + int text_w, text_h; + ltk_text_line_get_size(button->tl, &text_w, &text_h); + int box_size = ltk_size_to_pixel(theme.box_size, LTK_CAST_WIDGET(button)->last_dpi); + int pad = ltk_size_to_pixel(theme.pad, LTK_CAST_WIDGET(button)->last_dpi); + button->widget.ideal_w = text_w + pad * 3 + box_size; + button->widget.ideal_h = MAX(text_h, box_size) + pad * 2; +} + +static void +ltk_checkbutton_recalc_ideal_size(ltk_widget *self) { + ltk_checkbutton *button = LTK_CAST_CHECKBUTTON(self); + int font_size = ltk_size_to_pixel(theme.font_size, self->last_dpi); + ltk_text_line_set_font_size(button->tl, font_size); + recalc_ideal_size(button); +} + +ltk_checkbutton * +ltk_checkbutton_create(ltk_window *window, const char *text, int checked) { + ltk_checkbutton *button = ltk_malloc(sizeof(ltk_checkbutton)); + ltk_fill_widget_defaults(LTK_CAST_WIDGET(button), window, &vtable, 0, 0); + button->checked = checked; + + button->tl = ltk_text_line_create_const_text_default( + theme.font, ltk_size_to_pixel(theme.font_size, button->widget.last_dpi), text, -1 + ); + recalc_ideal_size(button); + button->widget.dirty = 1; + + return button; +} + +static void +ltk_checkbutton_destroy(ltk_widget *self, int shallow) { + (void)shallow; + ltk_checkbutton *button = LTK_CAST_CHECKBUTTON(self); + if (!button) { + ltk_warn("Tried to destroy NULL checkbutton.\n"); + return; + } + ltk_text_line_destroy(button->tl); + ltk_free(button); +} diff --git a/src/ltk/checkbutton.h b/src/ltk/checkbutton.h @@ -0,0 +1,37 @@ +/* + * Copyright (c) 2024 lumidify <nobody@lumidify.org> + * + * Permission to use, copy, modify, and/or distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +#ifndef LTK_CHECKBUTTON_H +#define LTK_CHECKBUTTON_H + +#include "text.h" +#include "widget.h" +#include "window.h" + +#define LTK_CHECKBUTTON_SIGNAL_CHANGED -1 +#define LTK_CHECKBUTTON_SIGNAL_INVALID -2 + +typedef struct { + ltk_widget widget; + ltk_text_line *tl; + int checked; +} ltk_checkbutton; + +ltk_checkbutton *ltk_checkbutton_create(ltk_window *window, const char *text, int checked); +int ltk_checkbutton_get_checked(ltk_checkbutton *button); +void ltk_checkbutton_set_checked(ltk_checkbutton *button, int checked); + +#endif /* LTK_CHECKBUTTON_H */ diff --git a/src/ltk/config.c b/src/ltk/config.c @@ -84,6 +84,7 @@ static struct theme_handlerinfo { {"theme:menuentry", &ltk_menuentry_get_theme_parseinfo, "theme:window", 0}, {"theme:submenu", &ltk_submenu_get_theme_parseinfo, "theme:window", 0}, {"theme:submenuentry", &ltk_submenuentry_get_theme_parseinfo, "theme:window", 0}, + {"theme:checkbutton", &ltk_checkbutton_get_theme_parseinfo, "theme:window", 0}, }; GEN_SORT_SEARCH_HELPERS(themehandler, struct theme_handlerinfo, name) diff --git a/src/ltk/event_xlib.c b/src/ltk/event_xlib.c @@ -335,12 +335,12 @@ next_event_base(ltk_renderdata *renderdata, ltk_renderwindow **windows, size_t n button = xevent.xbutton.button; /* FIXME: are the buttons really always defined as exactly these values? */ if (button >= 1 && button <= 3) { - if (xevent.xbutton.time - last_button_press[button] <= DOUBLECLICK_TIME && - DISTSQ(press_pos[button].x, press_pos[button].y, xevent.xbutton.x, xevent.xbutton.y) <= DOUBLECLICK_DISTSQ) { - if (was_2press[button]) { + if (xevent.xbutton.time - last_button_press[button - 1] <= DOUBLECLICK_TIME && + DISTSQ(press_pos[button - 1].x, press_pos[button - 1].y, xevent.xbutton.x, xevent.xbutton.y) <= DOUBLECLICK_DISTSQ) { + if (was_2press[button - 1]) { /* reset so normal press is sent again next time */ - was_2press[button] = 0; - last_button_press[button] = 0; + was_2press[button - 1] = 0; + last_button_press[button - 1] = 0; ltk_array_append_event(local_event_stack, (ltk_event){.button = { .type = LTK_3BUTTONPRESS_EVENT, .window_id = window_id, @@ -349,8 +349,8 @@ next_event_base(ltk_renderdata *renderdata, ltk_renderwindow **windows, size_t n .y = xevent.xbutton.y }}); } else { - was_2press[button] = 1; - last_button_press[button] = xevent.xbutton.time; + was_2press[button - 1] = 1; + last_button_press[button - 1] = xevent.xbutton.time; ltk_array_append_event(local_event_stack, (ltk_event){.button = { .type = LTK_2BUTTONPRESS_EVENT, .window_id = window_id, @@ -360,8 +360,8 @@ next_event_base(ltk_renderdata *renderdata, ltk_renderwindow **windows, size_t n }}); } } else { - last_button_press[button] = xevent.xbutton.time; - was_2press[button] = 0; + last_button_press[button - 1] = xevent.xbutton.time; + was_2press[button - 1] = 0; } *event = (ltk_event){.button = { .type = LTK_BUTTONPRESS_EVENT, @@ -370,8 +370,8 @@ next_event_base(ltk_renderdata *renderdata, ltk_renderwindow **windows, size_t n .x = xevent.xbutton.x, .y = xevent.xbutton.y }}; - press_pos[button].x = xevent.xbutton.x; - press_pos[button].y = xevent.xbutton.y; + press_pos[button - 1].x = xevent.xbutton.x; + press_pos[button - 1].y = xevent.xbutton.y; } else if (button >= 4 && button <= 7) { /* FIXME: compress multiple scroll events into one */ *event = (ltk_event){.scroll = { @@ -404,12 +404,12 @@ next_event_base(ltk_renderdata *renderdata, ltk_renderwindow **windows, size_t n ltk_assert(window_id < num_windows); button = xevent.xbutton.button; if (button >= 1 && button <= 3) { - if (xevent.xbutton.time - last_button_release[button] <= DOUBLECLICK_TIME && - DISTSQ(release_pos[button].x, release_pos[button].y, xevent.xbutton.x, xevent.xbutton.y) <= DOUBLECLICK_DISTSQ) { - if (was_2release[button]) { + if (xevent.xbutton.time - last_button_release[button - 1] <= DOUBLECLICK_TIME && + DISTSQ(release_pos[button - 1].x, release_pos[button - 1].y, xevent.xbutton.x, xevent.xbutton.y) <= DOUBLECLICK_DISTSQ) { + if (was_2release[button - 1]) { /* reset so normal release is sent again next time */ - was_2release[button] = 0; - last_button_release[button] = 0; + was_2release[button - 1] = 0; + last_button_release[button - 1] = 0; ltk_array_append_event(local_event_stack, (ltk_event){.button = { .type = LTK_3BUTTONRELEASE_EVENT, .window_id = window_id, @@ -418,8 +418,8 @@ next_event_base(ltk_renderdata *renderdata, ltk_renderwindow **windows, size_t n .y = xevent.xbutton.y }}); } else { - was_2release[button] = 1; - last_button_release[button] = xevent.xbutton.time; + was_2release[button - 1] = 1; + last_button_release[button - 1] = xevent.xbutton.time; ltk_array_append_event(local_event_stack, (ltk_event){.button = { .type = LTK_2BUTTONRELEASE_EVENT, .window_id = window_id, @@ -429,8 +429,8 @@ next_event_base(ltk_renderdata *renderdata, ltk_renderwindow **windows, size_t n }}); } } else { - last_button_release[button] = xevent.xbutton.time; - was_2release[button] = 0; + last_button_release[button - 1] = xevent.xbutton.time; + was_2release[button - 1] = 0; } *event = (ltk_event){.button = { .type = LTK_BUTTONRELEASE_EVENT, @@ -439,8 +439,8 @@ next_event_base(ltk_renderdata *renderdata, ltk_renderwindow **windows, size_t n .x = xevent.xbutton.x, .y = xevent.xbutton.y }}; - release_pos[button].x = xevent.xbutton.x; - release_pos[button].y = xevent.xbutton.y; + release_pos[button - 1].x = xevent.xbutton.x; + release_pos[button - 1].y = xevent.xbutton.y; } else { return 2; } diff --git a/src/ltk/widget.h b/src/ltk/widget.h @@ -45,6 +45,7 @@ typedef enum { LTK_WIDGET_IMAGE, LTK_WIDGET_WINDOW, LTK_WIDGET_SCROLLBAR, + LTK_WIDGET_CHECKBUTTON, LTK_NUM_WIDGETS, } ltk_widget_type; @@ -184,6 +185,7 @@ typedef struct { #define LTK_CAST_MENUENTRY(w) (ltk_assert(w->vtable->type == LTK_WIDGET_MENUENTRY), (ltk_menuentry *)(w)) #define LTK_CAST_SCROLLBAR(w) (ltk_assert(w->vtable->type == LTK_WIDGET_SCROLLBAR), (ltk_scrollbar *)(w)) #define LTK_CAST_BOX(w) (ltk_assert(w->vtable->type == LTK_WIDGET_BOX), (ltk_box *)(w)) +#define LTK_CAST_CHECKBUTTON(w) (ltk_assert(w->vtable->type == LTK_WIDGET_CHECKBUTTON), (ltk_checkbutton *)(w)) /* FIXME: a bit weird because window never gets some of these signals */ #define LTK_WIDGET_SIGNAL_KEY_PRESS 1 diff --git a/src/ltk/widget_internal.h b/src/ltk/widget_internal.h @@ -26,6 +26,7 @@ void ltk_window_get_theme_parseinfo(ltk_theme_parseinfo **p, size_t *len); void ltk_button_get_theme_parseinfo(ltk_theme_parseinfo **p, size_t *len); +void ltk_checkbutton_get_theme_parseinfo(ltk_theme_parseinfo **p, size_t *len); void ltk_label_get_theme_parseinfo(ltk_theme_parseinfo **p, size_t *len); void ltk_menu_get_theme_parseinfo(ltk_theme_parseinfo **p, size_t *len); void ltk_menuentry_get_theme_parseinfo(ltk_theme_parseinfo **p, size_t *len);