ui/motif/toolkit.c

Wed, 08 Jan 2025 20:35:24 +0100

author
Olaf Wintermann <olaf.wintermann@gmail.com>
date
Wed, 08 Jan 2025 20:35:24 +0100
changeset 441
752bd110375e
parent 433
605bb5dc34f1
permissions
-rw-r--r--

add ui.gtk.window.showtitle property for configuring the gtk headerbar show_title property

/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright 2024 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 <unistd.h>
#include <pthread.h>

#include "toolkit.h"
#include "toolbar.h"
#include "container.h"
#include "stock.h"
#include "../common/menu.h"
#include "../common/toolbar.h"
#include "../common/document.h"
#include "../common/properties.h"
#include <cx/buffer.h>

#include <X11/Intrinsic.h>
#include <Xm/CutPaste.h>

static XtAppContext app;
static Display *display;
static Widget active_window;
static const char *application_name;

static ui_callback   startup_func;
static void          *startup_data;
static ui_callback   open_func;
void                 *open_data;
static ui_callback   exit_func;
void                 *exit_data;

static ui_callback appclose_fnc;
static void *appclose_udata;

static int is_toplevel_realized = 0;

int event_pipe[2];


static String fallback[] = {
    	//"*fontList: -dt-interface system-medium-r-normal-s*utf*:",                 
        "*renderTable: rt",
        "*rt*fontType: FONT_IS_XFT",
        "*rt*fontName: Sans",
        "*rt*fontSize: 11",
        
        "*progresss_spinner*renderTable*fontType: FONT_IS_FONT",
        "*progresss_spinner*renderTable*fontName: Cursor",
        
        "*window_frame.shadowType: SHADOW_ETCHED_OUT",
        "*window_frame.shadowThickness: 1",
        "*togglebutton.shadowThickness: 1",
        "*togglebutton.highlightThickness: 2",
	NULL
};

void input_proc(XtPointer data, int *source, XtInputId *iid) {
    void *ptr;
    read(event_pipe[0], &ptr, sizeof(void*));
}

void ui_init(const char *appname, int argc, char **argv) { 
    application_name = appname;
    uic_init_global_context();
    
    XtToolkitInitialize();
    XtSetLanguageProc(NULL, NULL, NULL);
    app = XtCreateApplicationContext();
    XtAppSetFallbackResources(app, fallback);
    
    display =  XtOpenDisplay(app, NULL, appname, appname, NULL, 0, &argc, argv);
    
    uic_docmgr_init();
    uic_menu_init();
    uic_toolbar_init();
    uic_load_app_properties();
    
    if(pipe(event_pipe)) {
        fprintf(stderr, "UiError: Cannot create event pipe\n");
        exit(-1);
    }
    XtAppAddInput(
            app,
            event_pipe[0],
            (XtPointer)XtInputReadMask,
            input_proc,
            NULL);
}

const char* ui_appname() {
    return application_name;
}

Display* ui_motif_get_display() {
    return display;
}

void ui_onstartup(ui_callback f, void *userdata) {
    startup_func = f;
    startup_data = userdata;
}

void ui_onopen(ui_callback f, void *userdata) {
    open_func = f;
    open_data = userdata;
}

void ui_onexit(ui_callback f, void *userdata) {
    exit_func = f;
    exit_data = userdata;
}

void ui_main() {
    if(startup_func) {
        startup_func(NULL, startup_data);
    }
    XtAppMainLoop(app);
    if(exit_func) {
        exit_func(NULL, exit_data);
    }
    uic_store_app_properties();
}

void ui_exit_mainloop() {
    XtAppSetExitFlag(app);
}

void ui_secondary_event_loop(int *loop) {
    while(*loop && !XtAppGetExitFlag(app)) {
        XEvent event;
        XtAppNextEvent(app, &event);
        XtDispatchEvent(&event);
    }
}

void ui_show(UiObject *obj) {
    uic_check_group_widgets(obj->ctx);
    XtRealizeWidget(obj->widget);
}

void ui_close(UiObject *obj) {
    
}


void ui_set_enabled(UIWIDGET widget, int enabled) {
    XtSetSensitive(widget, enabled);
}

void ui_set_show_all(UIWIDGET widget, int value) {
    if(!value) {
        XtUnmanageChild(widget);
    }
}

void ui_set_visible(UIWIDGET widget, int visible) {
    if(visible) {
        XtManageChild(widget);
    } else {
        XtUnmanageChild(widget);
    }
}

static Boolean ui_job_finished(void *data) {
    UiJob *job = data;
    if(job->finish_callback) {
        UiEvent event;
        event.obj = job->obj;
        event.window = job->obj->window;
        event.document = job->obj->ctx->document;
        event.intval = 0;
        event.eventdata = NULL;
        job->finish_callback(&event, job->finish_data);
    }
    free(job);
    return TRUE;
}

static void* ui_jobthread(void *data) {
    UiJob *job = data;
    int result = job->job_func(job->job_data);
    if(!result) {
        write(event_pipe[1], &job, sizeof(void*)); // hack
        XtAppAddWorkProc(app, ui_job_finished, job);
        
    }
    return NULL;
}

void ui_job(UiObject *obj, ui_threadfunc tf, void *td, ui_callback f, void *fd) {
    UiJob *job = malloc(sizeof(UiJob));
    job->obj = obj;
    job->job_func = tf;
    job->job_data = td;
    job->finish_callback = f;
    job->finish_data = fd;
    pthread_t pid;
    pthread_create(&pid, NULL, ui_jobthread, job);
}

void ui_clipboard_set(char *str) {
    int length = strlen(str) + 1;
    
    Display *dp = XtDisplayOfObject(active_window);
    Window window = XtWindowOfObject(active_window);
    
    XmString label = XmStringCreateLocalized("toolkit_clipboard");
    long id = 0;
    
    while(XmClipboardStartCopy(
            dp,
            window,
            label,
            CurrentTime,
            NULL,
            NULL,
            &id) == ClipboardLocked);
    XmStringFree(label);
    
    while(XmClipboardCopy(
            dp,
            window,
            id,
            "STRING",
            str, 
            length,
            1,
            NULL) == ClipboardLocked);
    
    while(XmClipboardEndCopy(dp, window, id) == ClipboardLocked);
}

char* ui_clipboard_get() {
    Display *dp = XtDisplayOfObject(active_window);
    Window window = XtWindowOfObject(active_window);
    
    long id;
    size_t size = 128;
    char *buf = malloc(size);
    
    int r;
    for(;;) {
        r = XmClipboardRetrieve(dp, window, "STRING", buf, size, NULL, &id);
        if(r == ClipboardSuccess) {
            break;
        } else if(r == ClipboardTruncate) {
            size *= 2;
            buf = realloc(buf, size);
        } else if(r == ClipboardNoData) {
            free(buf);
            buf = NULL;
            break;
        }
    }
    
    return buf;
}

void ui_set_active_window(Widget w) {
    active_window = w;
}

Widget ui_get_active_window() {
    return active_window;
}

/*
 * doesn't work with gnome anymore
 */
void ui_window_dark_theme(Display *dp, Window window) {
    Atom atom = XInternAtom(dp, "_GTK_THEME_VARIANT", False);
    Atom type = XInternAtom(dp, "UTF8_STRING", False);
    XChangeProperty(
            dp, 
            window, 
            atom,
            type,
            8,
            PropModeReplace,
            (const unsigned char*)"dark",
            4);
}

void ui_destroy_eventdata(Widget w, XtPointer data, XtPointer d) {
    free(data);
}

void ui_set_widget_groups(UiContext *ctx, Widget widget, const int *groups) {
    if(!groups) {
        return;
    }
    size_t ngroups = uic_group_array_size(groups);
    ui_set_widget_ngroups(ctx, widget, groups, ngroups);
}

void ui_set_widget_ngroups(UiContext *ctx, Widget widget, const int *groups, size_t ngroups) {
    if(ngroups > 0) {
        uic_add_group_widget_i(ctx, widget, (ui_enablefunc)ui_set_enabled, groups, ngroups);
        ui_set_enabled(widget, FALSE);
    }
}

mercurial