ui/common/document.c

Sat, 10 May 2014 15:43:22 +0200

author
Olaf Wintermann <olaf.wintermann@gmail.com>
date
Sat, 10 May 2014 15:43:22 +0200
changeset 33
458831c574f4
parent 3
c1a75454b444
child 37
56016468753d
permissions
-rw-r--r--

added listview, sidebar and toolbar image button (GTK)

/*
 * 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 "document.h"

static UcxMap *documents;

void uic_docmgr_init() {
    documents = ucx_map_new(32);
}

void ui_set_document(UiObject *obj, void *document) {
    UiDocument *doc = ucx_map_get(documents, ucx_key(&document, sizeof(void*)));
    if(!doc) {
        return;
    }
    doc->obj = obj;
    
    if(obj->document) {
        ui_detach_document(obj, obj->document);
    }
    obj->document = document;
    
    UcxMapIterator i = ucx_map_iterator(doc->vars);
    UiVar *var;
    UCX_MAP_FOREACH(key, var, i) {
        UiVar *v = ucx_map_get(obj->ctx->vars, key);
        if(v) {
            if(v->isextern) {
                fprintf(
                        stderr,
                        "UI Error: external variable cannot be moved\n");
                return;
            }
            // check type
            if(var->type != v->type) {
                fprintf(stderr, "UI Error: var has incompatible type.\n");
                return;
            }
            
            // copy value
            uic_var_move(v, var, 1);
            var->from = v->from;
            
            // TODO: free var struct
            ucx_map_remove(obj->ctx->vars, key);
        }
    }
    
}

void ui_detach_document(UiObject *obj, void *document) {
    UiDocument *doc = ucx_map_get(documents, ucx_key(&document, sizeof(void*)));
    if(!doc) {
        return;
    }
    
    UcxMapIterator i = ucx_map_iterator(doc->vars);
    UiVar *var;
    UCX_MAP_FOREACH(key, var, i) {
        if(var->from && var->from != doc->vars) {
            // this var is bind to an outer widget, so we move it          
            UcxAllocator *a = var->from->allocator;
            UiVar *newvar = a->malloc(a->pool, sizeof(UiVar));
            newvar->value = uic_create_value(a, var->type);
            uic_var_move(var, newvar, 0);
            newvar->type = var->type;
            newvar->from = var->from;
            newvar->isextern = 0;
            
            ucx_map_put(var->from, key, newvar);
            
            //ucx_map_remove(doc->vars, key); // TODO: dont remove!
        }
    }
    
    doc->obj->document = NULL;
    doc->obj = NULL;
}

void* ui_document_new(size_t size) {
    UcxMempool *mp = ucx_mempool_new(256);
    UiDocument *uidoc = ucx_mempool_malloc(mp, sizeof(UiDocument));
    uidoc->obj = NULL;
    uidoc->mempool = mp;
    uidoc->vars = ucx_map_new_a(mp->allocator, 16);
    uidoc->subdocument = NULL;
    
    void *document = ucx_mempool_calloc(mp, 1, size);
    ucx_map_put(documents, ucx_key(&document, sizeof(void*)), uidoc);
    return document;
}

void ui_document_destroy(void *doc) {
    // TODO
}

void* ui_document_malloc(void *doc, size_t size) {
    UiDocument *uidoc = ucx_map_get(documents, ucx_key(&doc, sizeof(void*)));
    return ucx_mempool_malloc(uidoc->mempool, size);
}

void* ui_document_calloc(void *doc, size_t nelem, size_t elsize) {
    UiDocument *uidoc = ucx_map_get(documents, ucx_key(&doc, sizeof(void*)));
    return ucx_mempool_calloc(uidoc->mempool, nelem, elsize);
}

void  ui_document_free(void *doc, void *ptr) {
    UiDocument *uidoc = ucx_map_get(documents, ucx_key(&doc, sizeof(void*)));
    ucx_mempool_free(uidoc->mempool, ptr);
}

void* ui_document_realloc(void *doc, void *ptr, size_t size) {
    UiDocument *uidoc = ucx_map_get(documents, ucx_key(&doc, sizeof(void*)));
    return ucx_mempool_realloc(uidoc->mempool, ptr, size);
}

UiDocument* uic_getdocument(void *doc) {
    return doc ? ucx_map_get(documents, ucx_key(&doc, sizeof(void*))) : NULL;
}

UiVar* uic_document_getvar(UiDocument *doc, char *name) {
    return doc ? ucx_map_cstr_get(doc->vars, name) : NULL;
}

void uic_document_addvar(void *doc, char *name, int type, size_t vs) {
    UiDocument *uidoc = ucx_map_get(documents, ucx_key(&doc, sizeof(void*)));
    if(!uidoc) {
        return;
    }
    
    UiVar *newvar = ucx_mempool_malloc(uidoc->mempool, sizeof(UiVar));
    newvar->isextern = 0;
    newvar->type = type;
    newvar->value = ucx_mempool_calloc(uidoc->mempool, 1, vs);
    newvar->from = NULL;
    
    uic_document_addvar_v(uidoc, name, newvar, type, vs);
}

void uic_document_addvar_v(
        UiDocument *uidoc,
        char *name,
        UiVar *newvar,
        int type,
        size_t vs)
{ 
    // if the window context has this variable, we remove it and put it to
    // the document vars
    UiVar *var = uidoc->obj ?
            ucx_map_cstr_get(uidoc->obj->ctx->vars, name) : NULL;
    if(var) {
        // external variables cannot be moved
        if(var->isextern) {
            fprintf(
                    stderr,
                    "UI Error: external variable %s "
                    "cannot be moved to the document.\n",
                    name);
            return;
        }
        
        // check type
        if(var->type != type) {
            fprintf(stderr, "UI Error: var %s has incompatible type.\n", name);
            return;
        }
        
        // override document var with window context var
        memcpy(newvar->value, var->value, vs);
        
        newvar->from = var->from;
        
        // TODO: free var struct
        ucx_map_cstr_remove(uidoc->obj->ctx->vars, name);
    }
    
    // finally, add the new variable to the document
    ucx_map_cstr_put(uidoc->vars, name, newvar);
}

void uic_document_regvar(
        void *doc,
        char *name, 
        int type,
        size_t vs,
        void *value)
{
    UiDocument *uidoc = ucx_map_get(documents, ucx_key(&doc, sizeof(void*)));
    if(!uidoc) {
        return;
    }
    
    UiVar *newvar = ucx_mempool_malloc(uidoc->mempool, sizeof(UiVar));
    newvar->isextern = 1;
    newvar->type = type;
    newvar->value = value;
    newvar->from = NULL;
    
    uic_document_addvar_v(uidoc, name, newvar, type, vs);
}

void uic_var_move(UiVar *from, UiVar *to, int set) {
    switch(from->type) {
        case UI_VAR_INTEGER: {
            //memcpy(to->value, from->value, sizeof(UiInteger));
            UiInteger *f = from->value;
            UiInteger *t = to->value;
            t->get = f->get;
            t->set = f->set;
            t->obj = f->obj; 
            if(set) {
                t->set(t, t->value);
            } else {
                //t->value = t->get(t);
                f->value = f->get(f);
                //t->value = 0;
            }
            break;
        }
        case UI_VAR_STRING: {
            // TODO
            break;
        }
        case UI_VAR_TEXT: {
            UiText *f = from->value;
            UiText *t = to->value;
            char *tvalue = t->value;
            memcpy(t, f, sizeof(UiText));
            if(set) {
                t->set(t, tvalue);
            } else {
                f->value = f->get(f);
            }
            break;
        }
        case UI_VAR_LIST: {
            UiListVar *f = from->value;
            UiListVar *t = to->value;
            UiList *list = t->listptr->list;
            UiObserver *observers = f->listptr->list->observers;
            t->listptr = f->listptr;
            if(set) {
                t->listptr->list = list;
                list->observers = observers;
                ui_notify(observers, list);
            }
            break;
        }
    }
}

void ui_document_addint(void *doc, char *name) {
    uic_document_addvar(doc, name, UI_VAR_INTEGER, sizeof(UiInteger));
}

void ui_document_regint(void *doc, char *name, UiInteger *i) {
    uic_document_regvar(doc, name, UI_VAR_INTEGER, sizeof(UiInteger), i);
}

void ui_document_regtext(void *doc, char *name, UiText *text) {
    uic_document_regvar(doc, name, UI_VAR_TEXT, sizeof(UiText), text);
}

void ui_document_reglist(void *doc, char *name, UiList *list) {
    UiListVar *lv = ui_document_malloc(doc, sizeof(UiListVar));
    UiListPtr *lp = ui_document_malloc(doc, sizeof(UiListPtr));
    lv->listptr = lp;
    lp->list = list;
    uic_document_regvar(doc, name, UI_VAR_LIST, sizeof(UiListPtr), lv);
}

mercurial