ui/gtk/button.c

Sun, 07 Dec 2025 16:02:47 +0100

author
Olaf Wintermann <olaf.wintermann@gmail.com>
date
Sun, 07 Dec 2025 16:02:47 +0100
changeset 973
aa39a986da78
parent 967
ff4a8d10307b
permissions
-rw-r--r--

fix gtk3 build

/*
 * 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 <cx/buffer.h>
#include <cx/json.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
}

GtkWidget* ui_create_button(
        UiObject *obj,
        const char *label,
        const char *icon,
        const char *tooltip,
        ui_callback onclick,
        void *userdata,
        int event_value,
        bool activate_event)
{
    GtkWidget *button = gtk_button_new_with_label(label);
    ui_button_set_icon_name(button, icon);
    if(tooltip) {
        gtk_widget_set_tooltip_text(button, tooltip);
    }
    
    if(onclick) {
        UiEventData *event = malloc(sizeof(UiEventData));
        event->obj = obj;
        event->userdata = userdata;
        event->callback = onclick;
        event->value = event_value;
        event->customdata = NULL;
        event->customint = 0;

        g_signal_connect(
                button,
                "clicked",
                G_CALLBACK(ui_button_clicked),
                event);
        g_signal_connect(
                button,
                "destroy",
                G_CALLBACK(ui_destroy_userdata),
                event);
        if(activate_event) {
            g_signal_connect(
                button,
                "activate",
                G_CALLBACK(ui_button_clicked),
                event);
        }
    }
    
    return button;
}

UIWIDGET ui_button_create(UiObject *obj, UiButtonArgs *args) {
    GtkWidget *button = ui_create_button(obj, args->label, args->icon, args->tooltip, args->onclick, args->onclickdata, 0, FALSE);
    ui_set_name_and_style(button, args->name, args->style_class);
    ui_set_widget_states(obj->ctx, button, args->states);
    UiContainerPrivate *ct = (UiContainerPrivate*)obj->container_end;
    UiLayout layout = UI_ARGS2LAYOUT(args);
    ct->add(ct, button, &layout);
    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.eventdatatype = 0;
    e.intval = event->value;
    e.set = ui_get_setop();
    event->callback(&e, event->userdata);
}

void ui_button_set_label(UIWIDGET button, const char *label) {
    gtk_button_set_label(GTK_BUTTON(button), label);
}

void ui_button_set_icon(UIWIDGET button, const char *icon) {
    ui_button_set_icon_name(button, icon);
}

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(void *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 = event->var->value;
    e.eventdatatype = UI_EVENT_DATA_INTEGER_VALUE;
    e.intval = i->get(i);
    e.set = ui_get_setop();
    
    ui_notify_evt(i->observers, &e);
}

static void ui_toggled_callback(GtkToggleButton *widget, UiEventData *event) {
    UiEvent e;
    e.obj = event->obj;
    e.window = event->obj->window;
    e.document = event->obj->ctx->document;
    e.eventdata = NULL;
    e.eventdatatype = 0;
    e.intval = gtk_toggle_button_get_active(widget);
    e.set = ui_get_setop();
    event->callback(&e, event->userdata);    
}

static void ui_togglebutton_enable_state_callback(GtkToggleButton *widget, UiEventData *event) {
    if(gtk_toggle_button_get_active(widget)) {
        ui_set_state(event->obj->ctx, event->value);
    } else {
        ui_unset_state(event->obj->ctx, event->value);
    }
}

void ui_setup_togglebutton(
        UiObject *obj,
        GtkWidget *togglebutton,
        const char *label,
        const char *icon,
        const char *tooltip,
        const char *varname,
        UiInteger *value,
        ui_callback onchange,
        void *onchangedata,
        int enable_state)
{
    if(label) {
        gtk_button_set_label(GTK_BUTTON(togglebutton), label);
    }
    ui_button_set_icon_name(togglebutton, icon);
    if(tooltip) {
        gtk_widget_set_tooltip_text(togglebutton, tooltip);
    }
    
    ui_bind_togglebutton(
            obj,
            togglebutton,
            ui_toggle_button_get,
            ui_toggle_button_set,
            varname,
            value,
            (ui_toggled_func)ui_toggled_callback,
            onchange,
            onchangedata,
            (ui_toggled_func)ui_togglebutton_enable_state_callback,
            enable_state
            );
}

void ui_bind_togglebutton(
        UiObject *obj,
        GtkWidget *widget,
        int64_t (*getfunc)(UiInteger*),
        void (*setfunc)(UiInteger*, int64_t),
        const char *varname,
        UiInteger *value,
        void (*toggled_callback)(void*, void*),
        ui_callback onchange,
        void *onchangedata,
        void (*enable_state_func)(void*, void*),
        int enable_state)
{
    UiVar* var = uic_widget_var(obj->ctx, obj->ctx, value, varname, UI_VAR_INTEGER);
    if (var) {
        UiInteger* value = (UiInteger*)var->value;
        value->obj = widget;
        value->get = getfunc;
        value->set = setfunc;
        
        UiVarEventData *event = malloc(sizeof(UiVarEventData));
        event->obj = obj;
        event->var = var;
        event->observers = NULL;
        event->callback = NULL;
        event->userdata = NULL;

        g_signal_connect(
                widget,
                "toggled",
                G_CALLBACK(ui_toggled_obs),
                event);
        g_signal_connect(
                widget,
                "destroy",
                G_CALLBACK(ui_destroy_vardata),
                event);
    }
    
    if(onchange) {
        UiEventData *event = malloc(sizeof(UiEventData));
        event->obj = obj;
        event->userdata = onchangedata;
        event->callback = onchange;
        event->value = 0;
        event->customdata = NULL;
        event->customint = 0;
        
        g_signal_connect(
                widget,
                "toggled",
                G_CALLBACK(toggled_callback),
                event);
        g_signal_connect(
                widget,
                "destroy",
                G_CALLBACK(ui_destroy_userdata),
                event);
    }
    
    if(enable_state > 0) {
        UiEventData *event = malloc(sizeof(UiEventData));
        event->obj = obj;
        event->userdata = NULL;
        event->callback = NULL;
        event->value = enable_state;
        event->customdata = NULL;
        event->customint = 0;
        
        g_signal_connect(
                widget,
                "toggled",
                G_CALLBACK(enable_state_func),
                event);
        g_signal_connect(
                widget,
                "destroy",
                G_CALLBACK(ui_destroy_userdata),
                event);
    }
}

static UIWIDGET togglebutton_create(UiObject *obj, GtkWidget *widget, UiToggleArgs *args) {
    ui_setup_togglebutton(
            obj,
            widget,
            args->label,
            args->icon,
            args->tooltip,
            args->varname,
            args->value,
            args->onchange,
            args->onchangedata,
            args->enable_state);
    ui_set_name_and_style(widget, args->name, args->style_class);
    ui_set_widget_states(obj->ctx, widget, args->states);
    
    UiContainerPrivate *ct = (UiContainerPrivate*)obj->container_end;
    UiLayout layout = UI_ARGS2LAYOUT(args);
    ct->add(ct, widget, &layout);
    
    return widget;
}

UIWIDGET ui_togglebutton_create(UiObject* obj, UiToggleArgs *args) {
    return togglebutton_create(obj, gtk_toggle_button_new(), args);
}

#if GTK_MAJOR_VERSION >= 4

int64_t ui_check_button_get(UiInteger *integer) {
    GtkCheckButton *button = integer->obj;
    integer->value = (int)gtk_check_button_get_active(button);
    return integer->value;
}

void ui_check_button_set(UiInteger *integer, int64_t value) {
    GtkCheckButton *button = integer->obj;
    integer->value = value;
    gtk_check_button_set_active(button, value != 0 ? TRUE : FALSE);
}

static void ui_checkbox_callback(GtkCheckButton *widget, UiEventData *event) {
    UiEvent e;
    e.obj = event->obj;
    e.window = event->obj->window;
    e.document = event->obj->ctx->document;
    e.eventdata = NULL;
    e.eventdatatype = 0;
    e.intval = gtk_check_button_get_active(widget);
    e.set = ui_get_setop();
    event->callback(&e, event->userdata);    
}

static void ui_checkbox_enable_state(GtkCheckButton *widget, UiEventData *event) {
    if(gtk_check_button_get_active(widget)) {
        ui_set_state(event->obj->ctx, event->value);
    } else {
        ui_unset_state(event->obj->ctx, event->value);
    }
}

UIWIDGET ui_checkbox_create(UiObject* obj, UiToggleArgs *args) { 
    GtkWidget *widget = gtk_check_button_new_with_label(args->label); 
    ui_bind_togglebutton(
            obj,
            widget,
            ui_check_button_get,
            ui_check_button_set,
            args->varname,
            args->value,
            (ui_toggled_func)ui_checkbox_callback,
            args->onchange,
            args->onchangedata,
            (ui_toggled_func)ui_checkbox_enable_state,
            args->enable_state);
    
    ui_set_name_and_style(widget, args->name, args->style_class);
    ui_set_widget_states(obj->ctx, widget, args->states);
    
    UiContainerPrivate *ct = (UiContainerPrivate*)obj->container_end;
    UiLayout layout = UI_ARGS2LAYOUT(args);
    ct->add(ct, widget, &layout);
    
    return widget;
}

#else
UIWIDGET ui_checkbox_create(UiObject* obj, UiToggleArgs *args) {
    return togglebutton_create(obj, gtk_check_button_new(), args);
}
#endif


#if GTK_MAJOR_VERSION >= 3

static void switch_changed(
        GObject *gobject,
        GParamSpec *pspec,
        UiVarEventData *event)
{
    GtkSwitch *sw = GTK_SWITCH (gobject);
    gboolean active = gtk_switch_get_active (sw);
    
    UiEvent e;
    e.obj = event->obj;
    e.document = e.obj->ctx->document;
    e.window = e.obj->window;
    e.eventdata = NULL;
    e.eventdatatype = 0;
    e.set = ui_get_setop();
    
    if(event->callback) {
        event->callback(&e, event->userdata);
    }
    if(event->var) {
        UiInteger *i = event->var->value;
        ui_notify_evt(i->observers, &e);
    }
}

UIWIDGET ui_switch_create(UiObject* obj, UiToggleArgs *args) {
    GtkWidget *widget = gtk_switch_new();
    ui_set_name_and_style(widget, args->name, args->style_class);
    ui_set_widget_states(obj->ctx, widget, args->states);
    
    UiVar* var = uic_widget_var(obj->ctx, obj->ctx, args->value, args->varname, UI_VAR_INTEGER);
    if(var) {
        UiInteger *value = var->value;
        value->obj = widget;
        value->get = ui_switch_get;
        value->set = ui_switch_set;
        
        if(value->value) {
            gtk_switch_set_active(GTK_SWITCH(widget), TRUE);
        }
        
        
    }
    
    UiVarEventData *event = malloc(sizeof(UiVarEventData));
    event->obj = obj;
    event->callback = args->onchange;
    event->userdata = args->onchangedata;
    event->var = var;
    event->observers = NULL;
    
    g_signal_connect(
            widget,
            "notify::active",
            G_CALLBACK(switch_changed),
            event);
    
    g_signal_connect(
            widget,
            "destroy",
            G_CALLBACK(ui_destroy_vardata),
            event);
    
    UiContainerPrivate *ct = (UiContainerPrivate*)obj->container_end;
    UiLayout layout = UI_ARGS2LAYOUT(args);
    ct->add(ct, widget, &layout);
    
    return widget;
}

int64_t ui_switch_get(UiInteger *value) {
    GtkSwitch *sw = GTK_SWITCH((GtkWidget*)value->obj);
    value->value = gtk_switch_get_active(sw);
    return value->value;
}

void ui_switch_set(UiInteger *value, int64_t i) {
    GtkSwitch *sw = GTK_SWITCH((GtkWidget*)value->obj);
    value->value = i;
    gtk_switch_set_active(sw, i);
}
    
#else

UIWIDGET ui_switch_create(UiObject* obj, UiToggleArgs *args) {
    return ui_checkbox_create(obj, args);
}

#endif

#if GTK_MAJOR_VERSION >= 4
#define RADIOBUTTON_NEW(group, label) gtk_check_button_new_with_label(label)
#define RADIOBUTTON_SET_GROUP(button, group) 
#define RADIOBUTTON_GET_GROUP(button) GTK_CHECK_BUTTON(button)
#define RADIOBUTTON_GET_ACTIVE(button) gtk_check_button_get_active(GTK_CHECK_BUTTON(button))
#else
#define RADIOBUTTON_NEW(group, label) gtk_radio_button_new_with_label(group, label)
#define RADIOBUTTON_SET_GROUP(button, group) /* noop */
#define RADIOBUTTON_GET_GROUP(button) gtk_radio_button_get_group(GTK_RADIO_BUTTON(button))
#define RADIOBUTTON_GET_ACTIVE(button) gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(button))
#endif

static void radiobutton_toggled(void *widget, UiEventData *event) {
    UiEvent e;
    e.obj = event->obj;
    e.window = event->obj->window;
    e.document = event->obj->ctx->document;
    e.eventdata = NULL;
    e.eventdatatype = 0;
    e.intval = RADIOBUTTON_GET_ACTIVE(widget);
    e.set = ui_get_setop();
    event->callback(&e, event->userdata);    
}

typedef struct UiRadioButtonData {
    UiInteger *value;
    UiVarEventData *eventdata;
    UiBool first;
} UiRadioButtonData;

static void destroy_radiobutton(GtkWidget *w, UiRadioButtonData *data) {
    if(data->first) {
        ui_destroy_vardata(w, data->eventdata);
        g_slist_free(data->value->obj);
        data->value->obj = NULL;
        data->value->get = NULL;
        data->value->set = NULL;
    } else {
        free(data->eventdata);
    }
    free(data);
}

UIWIDGET ui_radiobutton_create(UiObject *obj, UiToggleArgs *args) {
    GSList *rg = NULL;
    UiInteger *rgroup;
    
    UiVar* var = uic_widget_var(obj->ctx, obj->ctx, args->value, args->varname, UI_VAR_INTEGER);
    
    UiBool first = FALSE;
    if(var) {
        rgroup = var->value;
        rg = rgroup->obj;
        if(!rg) {
            first = TRUE;
        }
    }
    
    GtkWidget *rbutton = RADIOBUTTON_NEW(rg, args->label); 
    ui_set_name_and_style(rbutton, args->name, args->style_class);
    ui_set_widget_states(obj->ctx, rbutton, args->states);
    if(rgroup) {
#if GTK_MAJOR_VERSION >= 4
        if(rg) {
            gtk_check_button_set_group(GTK_CHECK_BUTTON(rbutton), rg->data);
        }
        rg = g_slist_prepend(rg, rbutton);
#else
        gtk_radio_button_set_group(GTK_RADIO_BUTTON(rbutton), rg);
        rg = gtk_radio_button_get_group(GTK_RADIO_BUTTON(rbutton));
#endif
        
        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;
        
        UiRadioButtonData *rbdata = malloc(sizeof(UiRadioButtonData));
        rbdata->value = rgroup;
        rbdata->eventdata = event;
        rbdata->first = first;
        
        g_signal_connect(
                rbutton,
                "toggled",
                G_CALLBACK(ui_radio_obs),
                event);
        g_signal_connect(
                rbutton,
                "destroy",
                G_CALLBACK(destroy_radiobutton),
                rbdata);
    }
    
    if(args->onchange) {
        UiEventData *event = malloc(sizeof(UiEventData));
        event->obj = obj;
        event->userdata = args->onchangedata;
        event->callback = args->onchange;
        event->value = 0;
        event->customdata = NULL;
        event->customint = 0;
        
        g_signal_connect(
                rbutton,
                "toggled",
                G_CALLBACK(radiobutton_toggled),
                event);
        g_signal_connect(
                rbutton,
                "destroy",
                G_CALLBACK(ui_destroy_userdata),
                event);
    }
    
    UiContainerPrivate *ct = (UiContainerPrivate*)obj->container_end;
    UiLayout layout = UI_ARGS2LAYOUT(args);
    ct->add(ct, rbutton, &layout);
    
    return rbutton;
}

void ui_radio_obs(GtkToggleButton *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.eventdatatype = 0;
    e.intval = i->get(i);
    
    ui_notify_evt(i->observers, &e);
}

#if GTK_MAJOR_VERSION >= 4
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_check_button_get_active(GTK_CHECK_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_check_button_set_active(GTK_CHECK_BUTTON(ls->data), TRUE);
            break;
        }
        ls = ls->next;
        j++;
    }
    
    value->value = i;
}
#else
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;
}
#endif


static void ui_destroy_linkbutton(GtkWidget *widget, UiLinkButton *data) {
    free(data->link);
    free(data);
}

static const char* linkbutton_get_uri(UiLinkButton *link) {
    if(link->type == UI_LINK_BUTTON) {
        return link->link;
    } else {
        return gtk_link_button_get_uri(GTK_LINK_BUTTON(link->widget));
    }
}

static void linkbutton_set_uri(UiLinkButton *link, const char *uri) {
    if(link->type == UI_LINK_BUTTON) {
        free(link->link);
        link->link = uri ? strdup(uri) : NULL;
    } else {
        gtk_link_button_set_uri(GTK_LINK_BUTTON(link->widget), uri);
    }
}

static gboolean linkbutton_get_visited(UiLinkButton *link) {
    if(link->type == UI_LINK_BUTTON) {
        return FALSE;
    } else {
        gtk_link_button_get_visited(GTK_LINK_BUTTON(link->widget));
    }
}

static void linkbutton_set_visited(UiLinkButton *link, gboolean visited) {
    if(link->type != UI_LINK_BUTTON) {
        gtk_link_button_set_visited(GTK_LINK_BUTTON(link->widget), visited);
    }
}

/*
 * Apply linkbutton settings from json. Expects jsonvalue to be a valid
 * json object.
 * 
 * {
 *     "label": "label text",
 *     "uri": "http://example.com",
 *     "visited": true
 * }
 * 
 */
static void linkbutton_apply_value(UiLinkButton *link, const char *jsonvalue) {
    CxJson json;
    cxJsonInit(&json, NULL);
    cxJsonFill(&json, jsonvalue);
    
    CxJsonValue *value;
    if(cxJsonNext(&json, &value) == CX_JSON_NO_ERROR) {
        if(cxJsonIsObject(value)) {
            CxJsonValue *label = cxJsonObjGet(value, "label");
            CxJsonValue *uri = cxJsonObjGet(value, "uri");
            CxJsonValue *visited = cxJsonObjGet(value, "visited");
            if(label) {
                gtk_button_set_label(GTK_BUTTON(link->widget), cxJsonIsString(label) ? cxJsonAsString(label) : NULL);
            }
            if(uri) {
                linkbutton_set_uri(link, cxJsonIsString(uri) ? cxJsonAsString(uri) : NULL);
                
            }
            if(visited) {
                linkbutton_set_visited(link, cxJsonIsBool(visited) ? cxJsonAsBool(visited) : FALSE);
            }
        }        
        cxJsonValueFree(value);
    }
    cxJsonDestroy(&json);
}

static char* create_linkbutton_jsonvalue(const char *label, const char *uri, gboolean include_null, gboolean visited, gboolean set_visited) {
    CxJsonValue *obj = cxJsonCreateObj(NULL);
    if(label) {
        cxJsonObjPutString(obj, CX_STR("label"), label);
    } else if(include_null) {
        cxJsonObjPutLiteral(obj, CX_STR("label"), CX_JSON_NULL);
    }
    
    if(uri) {
        cxJsonObjPutString(obj, CX_STR("uri"), uri);
    } else if(include_null) {
        cxJsonObjPutLiteral(obj, CX_STR("uri"), CX_JSON_NULL);
    }
    
    if(set_visited) {
        cxJsonObjPutLiteral(obj, CX_STR("visited"), visited ? CX_JSON_TRUE : CX_JSON_FALSE);
    }
    
    CxJsonWriter writer = cxJsonWriterCompact();
    CxBuffer buf;
    cxBufferInit(&buf, NULL, 128, NULL, CX_BUFFER_AUTO_EXTEND);
    cxJsonWrite(&buf, obj, (cx_write_func)cxBufferWrite, &writer);
    cxJsonValueFree(obj);
    cxBufferTerminate(&buf);
    
    return buf.space;
}

static char* linkbutton_get_value(UiLinkButton *link) {
    const char *label = gtk_button_get_label(GTK_BUTTON(link->widget));
    const char *uri = linkbutton_get_uri(link);
    gboolean visited = linkbutton_get_visited(link);
    return create_linkbutton_jsonvalue(label, uri, TRUE, visited, TRUE);
}

static void linkbutton_callback(GtkWidget *widget, UiLinkButton *data) {
    if(data->onclick) {
        UiEvent e;
        e.obj = data->obj;
        e.document = e.obj->ctx->document;
        e.window = e.obj->window;
        e.eventdata = (char*)linkbutton_get_uri(data);
        e.eventdatatype = UI_EVENT_DATA_STRING;
        e.intval = 0;
        e.set = ui_get_setop();
        data->onclick(&e, data->onclickdata);
    }
}

static void linkbutton_clicked(GtkWidget *widget, UiLinkButton *data) {
    linkbutton_callback(widget, data);
    if(data->link) {   
#if GTK_CHECK_VERSION(4, 0, 0)
        GtkUriLauncher *launcher = gtk_uri_launcher_new (data->link);
        gtk_uri_launcher_launch (launcher, NULL, NULL, NULL, NULL);
        g_object_unref (launcher);
#elif GTK_CHECK_VERSION(3, 22, 0)
        GError *error = NULL;
        gtk_show_uri_on_window(NULL, data->link, GDK_CURRENT_TIME, &error);
#elif
        // TODO: call xdg-open
#endif
    }
}

static gboolean linkbutton_activate_link(GtkLinkButton *self, UiLinkButton *data) {
    linkbutton_callback(data->widget, data);
    return data->nofollow;
}

UIWIDGET ui_linkbutton_create(UiObject *obj, UiLinkButtonArgs *args) {
    UiLinkButton *data = malloc(sizeof(UiLinkButton));
    memset(data, 0, sizeof(UiLinkButton));
    data->obj = obj;
    data->type = args->type;
    data->nofollow = args->nofollow;
    data->onclick = args->onclick;
    data->onclickdata = args->onclickdata;
    
    GtkWidget *button;
    if(args->type == UI_LINK_BUTTON) {
        button = gtk_button_new();
        g_signal_connect(
                button,
                "clicked",
                G_CALLBACK(linkbutton_clicked),
                data);
    } else {
        button = gtk_link_button_new("file:///");
        g_signal_connect(
                button,
                "activate-link",
                G_CALLBACK(linkbutton_activate_link),
                data);
    }
    gtk_button_set_label(GTK_BUTTON(button), args->label);
#if GTK_CHECK_VERSION(4, 0, 0)
    gtk_button_set_can_shrink(GTK_BUTTON(button), TRUE);
#elif GTK_MAJOR_VERSION == 3
    GtkWidget *child = gtk_bin_get_child(GTK_BIN(button));
    gtk_label_set_ellipsize(GTK_LABEL(child), PANGO_ELLIPSIZE_END);
#endif
    g_object_set_data(G_OBJECT(button), "ui_linkbutton", data);
    g_signal_connect(
            button,
            "destroy",
            G_CALLBACK(ui_destroy_linkbutton),
            data);
    
    data->widget = button;
    
    UiVar *var = uic_widget_var(obj->ctx, obj->ctx, args->value, args->varname, UI_VAR_STRING);
    if(var) {
        UiString *str = var->value;
        char *current_value = ui_get(str);
        if(current_value) {
            linkbutton_apply_value(data, current_value);
        }
        
        str->obj = data;
        str->get = ui_linkbutton_get;
        str->set = ui_linkbutton_set;
    }
    
    ui_set_name_and_style(button, args->name, args->style_class);
    ui_set_widget_states(obj->ctx, button, args->states);
    UiContainerPrivate *ct = (UiContainerPrivate*)obj->container_end;
    UiLayout layout = UI_ARGS2LAYOUT(args);
    ct->add(ct, button, &layout);
    
    return button;
}

char* ui_linkbutton_get(UiString *s) {
    UiLinkButton *link = s->obj;
    if(s->value.free) {
        s->value.free(s->value.ptr);
    }
    s->value.ptr = linkbutton_get_value(link);
    s->value.free = free;
    return s->value.ptr;
}

void ui_linkbutton_set(UiString *s, const char *str) {
    linkbutton_apply_value(s->obj, str);
    if(s->value.free) {
        s->value.free(s->value.ptr);
    }
#if GTK_MAJOR_VERSION == 3
    UiLinkButton *data = s->obj;
    GtkWidget *child = gtk_bin_get_child(GTK_BIN(data->widget));
    gtk_label_set_ellipsize(GTK_LABEL(child), PANGO_ELLIPSIZE_END);
#endif
}


void ui_linkbutton_value_set(UiString *str, const char *label, const char *uri) {
    char *value = create_linkbutton_jsonvalue(label, uri, TRUE, FALSE, TRUE);
    ui_set(str, value);
    free(value);
}

void ui_linkbutton_value_set_label(UiString *str, const char *label) {
    char *value = create_linkbutton_jsonvalue(label, NULL, FALSE, FALSE, TRUE);
    ui_set(str, value);
    free(value);
}

void ui_linkbutton_value_set_uri(UiString *str, const char *uri) {
    char *value = create_linkbutton_jsonvalue(NULL, uri, FALSE, FALSE, TRUE);
    ui_set(str, value);
    free(value);
}

void ui_linkbutton_value_set_visited(UiString *str, UiBool visited) {
    char *value = create_linkbutton_jsonvalue(NULL, NULL, FALSE, visited, TRUE);
    ui_set(str, value);
    free(value);
}


void ui_linkbutton_set_label(UIWIDGET button, const char *label) {
    gtk_button_set_label(GTK_BUTTON(button), label);
}

void ui_linkbutton_set_uri(UIWIDGET button, const char *label) {
    UiLinkButton *link = g_object_get_data(G_OBJECT(button), "ui_linkbutton");
    if(link) {
        linkbutton_set_uri(link, label);
    } else {
        fprintf(stderr, "Error: ui_linkbutton_set_label: widget is not a linkbutton\n");
    }
}

void ui_linkbutton_set_visited(UIWIDGET button, UiBool visited) {
    UiLinkButton *link = g_object_get_data(G_OBJECT(button), "ui_linkbutton");
    if(link) {
        linkbutton_set_visited(link, visited);
    } else {
        fprintf(stderr, "Error: ui_linkbutton_set_label: widget is not a linkbutton\n");
    }
}

char* ui_linkbutton_get_label(UIWIDGET button) {
    const char *label = gtk_button_get_label(GTK_BUTTON(button));
    return label ? strdup(label) : NULL;
}

char* ui_linkbutton_get_uri(UIWIDGET button) {
    UiLinkButton *link = g_object_get_data(G_OBJECT(button), "ui_linkbutton");
    if(link) {
        const char *uri = linkbutton_get_uri(link);
        return uri ? strdup(uri) : NULL;
    } else {
        fprintf(stderr, "Error: ui_linkbutton_set_label: widget is not a linkbutton\n");
    }
    return NULL;
}

UiBool ui_linkbutton_get_visited(UIWIDGET button) {
    UiLinkButton *link = g_object_get_data(G_OBJECT(button), "ui_linkbutton");
    if(link) {
        return linkbutton_get_visited(link);
    } else {
        fprintf(stderr, "Error: ui_linkbutton_set_label: widget is not a linkbutton\n");
    }
    return FALSE;
}

mercurial