ui/gtk/toolbar.c

Wed, 21 Jan 2015 20:38:21 +0100

author
Olaf Wintermann <olaf.wintermann@gmail.com>
date
Wed, 21 Jan 2015 20:38:21 +0100
changeset 77
bc0ed99e49c7
parent 49
a80ba8741be6
child 86
3c63f57a8f77
permissions
-rw-r--r--

fixed memory allocation bug

/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright 2014 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 <string.h>

#include "toolbar.h"
#include "button.h"
#include "image.h"
#include "../../ucx/mempool.h"
#include "../common/context.h"

static UcxMap *toolbar_items;
static UcxList *defaults;

void ui_toolbar_init() {
    toolbar_items = ucx_map_new(16);
}

void ui_toolitem(char *name, char *label, ui_callback f, void *udata) {
    ui_toolitem_img(name, label, NULL, f, udata);
}

void ui_toolitem_st(char *name, char *stockid, ui_callback f, void *userdata) {
    ui_toolitem_stgr(name, stockid, f, userdata, -1);
}

void ui_toolitem_sti(char *name, char *stockid, ui_callback f, void *userdata) {
    ui_toolitem_stgri(name, stockid, f, userdata, -1);
}

void ui_toolitem_stgr(char *name, char *stockid, ui_callback f, void *userdata, ...) {
    va_list ap;
    va_start(ap, userdata);
    ui_toolitem_vstgr(name, stockid, 0, f, userdata, ap);
    va_end(ap);
}

void ui_toolitem_stgri(char *name, char *stockid, ui_callback f, void *userdata, ...) {
    va_list ap;
    va_start(ap, userdata);
    ui_toolitem_vstgr(name, stockid, 1, f, userdata, ap);
    va_end(ap);
}

void ui_toolitem_img(char *name, char *label, char *img, ui_callback f, void *udata) {
    UiToolItem *item = malloc(sizeof(UiToolItem));
    item->item.add_to = (ui_toolbar_add_f)add_toolitem_widget;
    item->label = label;
    item->image = img;
    item->callback = f;
    item->userdata = udata;
    item->isimportant = 0;
    item->groups = NULL;
    
    ucx_map_cstr_put(toolbar_items, name, item);
}

void ui_toolitem_vstgr(
        char *name,
        char *stockid,
        int isimportant,
        ui_callback f,
        void *userdata,
        va_list ap)
{
    UiStToolItem *item = malloc(sizeof(UiStToolItem));
    item->item.add_to = (ui_toolbar_add_f)add_toolitem_st_widget;
    item->stockid = stockid;
    item->callback = f;
    item->userdata = userdata;
    item->groups = NULL;
    item->isimportant = isimportant;
    
    // add groups
    int group;
    while((group = va_arg(ap, int)) != -1) {
        item->groups = ucx_list_append(item->groups, (void*)(intptr_t)group);
    }
    
    ucx_map_cstr_put(toolbar_items, name, item);
}

void ui_toolitem_toggle_st(char *name, char *stockid, ui_callback f, void *udata) {
    ui_toolitem_toggle_stgr(name, stockid, f, udata, -1);
}

void ui_toolitem_toggle_stgr(char *name, char *stockid, ui_callback f, void *udata, ...) {
    UiStToolItem *item = malloc(sizeof(UiStToolItem));
    item->item.add_to = (ui_toolbar_add_f)add_toolitem_toggle_st_widget;
    item->stockid = stockid;
    item->callback = f;
    item->userdata = udata;
    item->groups = NULL;
    item->isimportant = 0;
    
    // add groups
    va_list ap;
    va_start(ap, udata);
    int group;
    while((group = va_arg(ap, int)) != -1) {
        item->groups = ucx_list_append(item->groups, (void*)(intptr_t)group);
    }
    va_end(ap);
    
    ucx_map_cstr_put(toolbar_items, name, item);
}

void ui_toolitem_toggle_imggr(char *name, char *label, char *img, ui_callback f, void *udata, ...) {
    UiToolItem *item = malloc(sizeof(UiToolItem));
    item->item.add_to = (ui_toolbar_add_f)add_toolitem_toggle_widget;
    item->label = label;
    item->image = img;
    item->callback = f;
    item->userdata = udata;
    item->groups = NULL;
    item->isimportant = 0;
    
    // add groups
    va_list ap;
    va_start(ap, udata);
    int group;
    while((group = va_arg(ap, int)) != -1) {
        item->groups = ucx_list_append(item->groups, (void*)(intptr_t)group);
    }
    va_end(ap);
    
    ucx_map_cstr_put(toolbar_items, name, item);
}


void ui_toolbar_add_default(char *name) {
    char *s = strdup(name);
    defaults = ucx_list_append(defaults, s);
}

GtkWidget* ui_create_toolbar(UiObject *obj) {
    if(!defaults) {
        return NULL;
    }
    
    GtkWidget *toolbar = gtk_toolbar_new();
#ifdef UI_GTK3
    gtk_style_context_add_class(
            gtk_widget_get_style_context(toolbar),
            GTK_STYLE_CLASS_PRIMARY_TOOLBAR);
#endif
    
    GtkToolbar *tb = GTK_TOOLBAR(toolbar);
    UCX_FOREACH(elm, defaults) {
        UiToolItemI *item = ucx_map_cstr_get(toolbar_items, elm->data);
        if(item) {
            item->add_to(tb, item, obj);
        } else if(!strcmp(elm->data, "@separator")) {
            gtk_toolbar_insert(tb, gtk_separator_tool_item_new(), -1);
        } else {
            fprintf(stderr, "UI Error: Unknown toolbar item: %s\n", elm->data);
        }
    }
    
    return toolbar;
}

void add_toolitem_widget(GtkToolbar *tb, UiToolItem *item, UiObject *obj) {
    GtkToolItem *button = gtk_tool_button_new(NULL, item->label);
    gtk_tool_item_set_homogeneous(button, FALSE);
    if(item->image) {
        GdkPixbuf *pixbuf = ui_get_image(item->image);
        GtkWidget *image = gtk_image_new_from_pixbuf(pixbuf);
        gtk_tool_button_set_icon_widget(GTK_TOOL_BUTTON(button), image);
    }
    
    if(item->callback) {
        UiEventData *event = ucx_mempool_malloc(
                obj->ctx->mempool,
                sizeof(UiEventData));
        event->obj = obj;
        event->userdata = item->userdata;
        event->callback = item->callback;
        
        g_signal_connect(
                button,
                "clicked",
                G_CALLBACK(ui_button_clicked),
                event);
    }
    
    gtk_toolbar_insert(tb, button, -1);
    
    if(item->groups) {
        uic_add_group_widget(obj->ctx, button, item->groups);
    }
}

void add_toolitem_st_widget(GtkToolbar *tb, UiStToolItem *item, UiObject *obj) {
    GtkToolItem *button = gtk_tool_button_new_from_stock(item->stockid);
    gtk_tool_item_set_homogeneous(button, FALSE);
    if(item->isimportant) {
        gtk_tool_item_set_is_important(button, TRUE);
    }
    
    if(item->callback) {
        UiEventData *event = ucx_mempool_malloc(
                obj->ctx->mempool,
                sizeof(UiEventData));
        event->obj = obj;
        event->userdata = item->userdata;
        event->callback = item->callback;
        
        g_signal_connect(
                button,
                "clicked",
                G_CALLBACK(ui_button_clicked),
                event);
    }
    
    gtk_toolbar_insert(tb, button, -1);
    
    if(item->groups) {
        uic_add_group_widget(obj->ctx, button, item->groups);
    }
}

void add_toolitem_toggle_widget(GtkToolbar *tb, UiToolItem *item, UiObject *obj) {
    GtkToolItem *button = gtk_toggle_tool_button_new();
    gtk_tool_item_set_homogeneous(button, FALSE);
    if(item->label) {
        gtk_tool_button_set_label(GTK_TOOL_BUTTON(button), item->label);
    }
    if(item->image) {
        GdkPixbuf *pixbuf = ui_get_image(item->image);
        GtkWidget *image = gtk_image_new_from_pixbuf(pixbuf);
        gtk_tool_button_set_icon_widget(GTK_TOOL_BUTTON(button), image);
    }    
    
    if(item->callback) {
        UiEventData *event = ucx_mempool_malloc(
                obj->ctx->mempool,
                sizeof(UiEventData));
        event->obj = obj;
        event->userdata = item->userdata;
        event->callback = item->callback;
        
        g_signal_connect(
                button,
                "toggled",
                G_CALLBACK(ui_button_toggled),
                event);
    }
    
    gtk_toolbar_insert(tb, button, -1);
    
    if(item->groups) {
        uic_add_group_widget(obj->ctx, button, item->groups);
    }
}

void add_toolitem_toggle_st_widget(GtkToolbar *tb, UiStToolItem *item, UiObject *obj) {
    GtkToolItem *button = gtk_toggle_tool_button_new_from_stock(item->stockid);
    gtk_tool_item_set_homogeneous(button, FALSE);
    
    if(item->callback) {
        UiEventData *event = ucx_mempool_malloc(
                obj->ctx->mempool,
                sizeof(UiEventData));
        event->obj = obj;
        event->userdata = item->userdata;
        event->callback = item->callback;
        
        g_signal_connect(
                button,
                "toggled",
                G_CALLBACK(ui_button_toggled),
                event);
    }
    
    gtk_toolbar_insert(tb, button, -1);
    
    if(item->groups) {
        uic_add_group_widget(obj->ctx, button, item->groups);
    }
}

mercurial