Tue, 11 Jun 2024 21:47:43 +0200
fix gtk executing ui_job_finished even if no finish callback was set
/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. * * Copyright 2017 Olaf Wintermann. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include <stdio.h> #include <stdlib.h> #include "button.h" #include "container.h" #include <cx/allocator.h> #include "../common/context.h" #include "../common/object.h" void ui_button_set_icon_name(GtkWidget *button, const char *icon) { if(!icon) { return; } #ifdef UI_GTK4 gtk_button_set_icon_name(GTK_BUTTON(button), icon); #else #if GTK_CHECK_VERSION(2, 6, 0) GtkWidget *image = gtk_image_new_from_icon_name(icon, GTK_ICON_SIZE_BUTTON); if(image) { gtk_button_set_image(GTK_BUTTON(button), image); } #else // TODO #endif #endif } UIWIDGET ui_button_create(UiObject *obj, UiButtonArgs args) { UiObject* current = uic_current_obj(obj); GtkWidget *button = gtk_button_new(); if(args.label) { gtk_button_set_label(GTK_BUTTON(button), args.label); } ui_button_set_icon_name(button, args.icon); if(args.onclick) { UiEventData *event = malloc(sizeof(UiEventData)); event->obj = obj; event->userdata = args.onclickdata; event->callback = args.onclick; event->value = 0; event->customdata = NULL; g_signal_connect( button, "clicked", G_CALLBACK(ui_button_clicked), event); g_signal_connect( button, "destroy", G_CALLBACK(ui_destroy_userdata), event); } UI_APPLY_LAYOUT1(current, args); current->container->add(current->container, button, FALSE); return button; } void ui_button_clicked(GtkWidget *widget, UiEventData *event) { UiEvent e; e.obj = event->obj; e.window = event->obj->window; e.document = event->obj->ctx->document; e.eventdata = NULL; e.intval = event->value; event->callback(&e, event->userdata); } int64_t ui_toggle_button_get(UiInteger *integer) { GtkToggleButton *button = integer->obj; integer->value = (int)gtk_toggle_button_get_active(button); return integer->value; } void ui_toggle_button_set(UiInteger *integer, int64_t value) { GtkToggleButton *button = integer->obj; integer->value = value; gtk_toggle_button_set_active(button, value != 0 ? TRUE : FALSE); } void ui_toggled_obs(GtkToggleToolButton *widget, UiVarEventData *event) { UiEvent e; e.obj = event->obj; e.window = event->obj->window; e.document = event->obj->ctx->document; e.eventdata = event->var->value; e.intval = gtk_toggle_tool_button_get_active(widget); UiInteger *i = event->var->value; ui_notify_evt(i->observers, &e); } static UIWIDGET togglebutton_create(UiObject *obj, GtkWidget *widget, UiToggleArgs args) { UiObject* current = uic_current_obj(obj); if(args.label) { gtk_button_set_label(GTK_BUTTON(widget), args.label); } ui_button_set_icon_name(widget, args.icon); UiVar* var = uic_widget_var(obj->ctx, current->ctx, args.value, args.varname, UI_VAR_INTEGER); if (var) { UiInteger* value = (UiInteger*)var->value; value->obj = widget; value->get = ui_toggle_button_get; value->set = ui_toggle_button_set; UiVarEventData *event = malloc(sizeof(UiVarEventData)); event->obj = obj; event->var = var; event->observers = NULL; event->callback = NULL; event->userdata = NULL; g_signal_connect( widget, "clicked", G_CALLBACK(ui_toggled_obs), event); g_signal_connect( widget, "destroy", G_CALLBACK(ui_destroy_vardata), event); } UI_APPLY_LAYOUT1(current, args); current->container->add(current->container, widget, FALSE); return widget; } UIWIDGET ui_togglebutton_create(UiObject* obj, UiToggleArgs args) { return togglebutton_create(obj, gtk_toggle_button_new(), args); } UIWIDGET ui_checkbox_create(UiObject* obj, UiToggleArgs args) { return togglebutton_create(obj, gtk_check_button_new(), args); } UIWIDGET ui_switch_create(UiObject* obj, UiToggleArgs args) { #ifdef UI_GTK3 return NULL; // TODO #else return ui_checkbox_create(obj, args); #endif } UIWIDGET ui_radiobutton_create(UiObject *obj, UiToggleArgs args) { UiObject* current = uic_current_obj(obj); GSList *rg = NULL; UiInteger *rgroup; UiVar* var = NULL; if (args.value) { var = uic_create_value_var(current->ctx, args.value); } else if (args.varname) { var = uic_create_var(obj->ctx, args.varname, UI_VAR_INTEGER); } UiBool first = FALSE; if(var) { rgroup = var->value; rg = rgroup->obj; if(!rg) { first = TRUE; } } GtkWidget *rbutton = gtk_radio_button_new_with_label(rg, args.label ? args.label : ""); rg = gtk_radio_button_get_group(GTK_RADIO_BUTTON(rbutton)); if(rgroup) { rgroup->obj = rg; rgroup->get = ui_radiobutton_get; rgroup->set = ui_radiobutton_set; ui_radiobutton_set(rgroup, rgroup->value); UiVarEventData *event = malloc(sizeof(UiVarEventData)); event->obj = obj; event->var = var; event->observers = NULL; event->callback = NULL; event->userdata = NULL; g_signal_connect( rbutton, "clicked", G_CALLBACK(ui_radio_obs), event); if(first) { g_signal_connect( rbutton, "destroy", G_CALLBACK(ui_destroy_vardata), event); } } UiContainer *ct = uic_get_current_container(obj); ct->add(ct, rbutton, FALSE); return rbutton; } void ui_radio_obs(GtkToggleToolButton *widget, UiVarEventData *event) { UiInteger *i = event->var->value; UiEvent e; e.obj = event->obj; e.window = event->obj->window; e.document = event->obj->ctx->document; e.eventdata = NULL; e.intval = i->get(i); ui_notify_evt(i->observers, &e); } int64_t ui_radiobutton_get(UiInteger *value) { int selection = 0; GSList *ls = value->obj; int i = 0; guint len = g_slist_length(ls); while(ls) { if(gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(ls->data))) { selection = len - i - 1; break; } ls = ls->next; i++; } value->value = selection; return selection; } void ui_radiobutton_set(UiInteger *value, int64_t i) { GSList *ls = value->obj; int s = g_slist_length(ls) - 1 - i; int j = 0; while(ls) { if(j == s) { gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(ls->data), TRUE); break; } ls = ls->next; j++; } value->value = i; }