HelenOS sources

root/uspace/app/taskbar/taskbar.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. taskbar_wnd_close
  2. taskbar_wnd_kbd
  3. taskbar_wnd_pos
  4. taskbar_create
  5. taskbar_destroy
  6. taskbar_notif_cb

/*
 * Copyright (c) 2024 Jiri Svoboda
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * - Redistributions of source code must retain the above copyright
 *   notice, this list of conditions and the following disclaimer.
 * - 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.
 * - The name of the author may not be used to endorse or promote products
 *   derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
 */

/** @addtogroup taskbar
 * @{
 */
/** @file Taskbar
 */

#include <gfx/coord.h>
#include <io/pos_event.h>
#include <stdio.h>
#include <stdlib.h>
#include <str.h>
#include <ui/fixed.h>
#include <ui/resource.h>
#include <ui/ui.h>
#include <ui/window.h>
#include <wndmgt.h>
#include "clock.h"
#include "taskbar.h"
#include "tbsmenu.h"
#include "wndlist.h"

#define TASKBAR_CONFIG_FILE "/cfg/taskbar.sif"

static void taskbar_wnd_close(ui_window_t *, void *);
static void taskbar_wnd_kbd(ui_window_t *, void *, kbd_event_t *);
static void taskbar_wnd_pos(ui_window_t *, void *, pos_event_t *);
static void taskbar_notif_cb(void *);

static ui_window_cb_t window_cb = {
        .close = taskbar_wnd_close,
        .kbd = taskbar_wnd_kbd,
        .pos = taskbar_wnd_pos
};

/** Window close button was clicked.
 *
 * @param window Window
 * @param arg Argument (taskbar)
 */
static void taskbar_wnd_close(ui_window_t *window, void *arg)
{
        taskbar_t *taskbar = (taskbar_t *) arg;

        ui_quit(taskbar->ui);
}

/** Window received keyboard event.
 *
 * @param window Window
 * @param arg Argument (taskbar)
 * @param event Keyboard event
 */
static void taskbar_wnd_kbd(ui_window_t *window, void *arg, kbd_event_t *event)
{
        taskbar_t *taskbar = (taskbar_t *) arg;
        ui_evclaim_t claim;

        /* Remember ID of device that sent the last event */
        taskbar->wndlist->ev_idev_id = event->kbd_id;
        taskbar->tbsmenu->ev_idev_id = event->kbd_id;

        claim = ui_window_def_kbd(window, event);
        if (claim == ui_claimed)
                return;

        if (event->type == KEY_PRESS && (event->mods & KM_CTRL) == 0 &&
            (event->mods & KM_ALT) == 0 && (event->mods & KM_SHIFT) == 0 &&
            event->key == KC_ENTER) {
                if (!tbsmenu_is_open(taskbar->tbsmenu))
                        tbsmenu_open(taskbar->tbsmenu);
        }
}

/** Window received position event.
 *
 * @param window Window
 * @param arg Argument (taskbar)
 * @param event Position event
 */
static void taskbar_wnd_pos(ui_window_t *window, void *arg, pos_event_t *event)
{
        taskbar_t *taskbar = (taskbar_t *) arg;

        /* Remember ID of device that sent the last event */
        taskbar->wndlist->ev_idev_id = event->pos_id;
        taskbar->tbsmenu->ev_idev_id = event->pos_id;

        ui_window_def_pos(window, event);
}

/** Create taskbar.
 *
 * @param display_spec Display specification
 * @param wndmgt_svc Window management service (or WNDMGT_DEFAULT)
 * @param rtaskbar Place to store pointer to new taskbar
 * @return @c EOK on success or an error coe
 */
errno_t taskbar_create(const char *display_spec, const char *wndmgt_svc,
    taskbar_t **rtaskbar)
{
        ui_wnd_params_t params;
        taskbar_t *taskbar = NULL;
        gfx_rect_t scr_rect;
        gfx_rect_t rect;
        char *dspec = NULL;
        char *qmark;
        errno_t rc;

        taskbar = calloc(1, sizeof(taskbar_t));
        if (taskbar == NULL) {
                printf("Out of memory.\n");
                rc = ENOMEM;
                goto error;
        }

        dspec = str_dup(display_spec);
        if (dspec == NULL) {
                printf("Out of memory.\n");
                rc = ENOMEM;
                goto error;
        }

        /* Remove additional arguments */
        qmark = str_chr(dspec, '?');
        if (qmark != NULL)
                *qmark = '\0';

        rc = ui_create(display_spec, &taskbar->ui);
        if (rc != EOK) {
                printf("Error creating UI on display %s.\n", display_spec);
                goto error;
        }

        rc = ui_get_rect(taskbar->ui, &scr_rect);
        if (rc != EOK) {
                if (str_cmp(display_spec, UI_DISPLAY_NULL) != 0) {
                        printf("Error getting screen dimensions.\n");
                        goto error;
                }

                /* For the sake of unit tests */
                scr_rect.p0.x = 0;
                scr_rect.p0.y = 0;
                scr_rect.p1.x = 100;
                scr_rect.p1.y = 100;
        }

        ui_wnd_params_init(&params);
        params.caption = "Taskbar";
        params.placement = ui_wnd_place_bottom_left;

        /* Window has no titlebar */
        params.style &= ~ui_wds_titlebar;

        /* Window is not obscured by other windows */
        params.flags |= ui_wndf_topmost;

        /* Prevent taskbar window from being listed in taskbar */
        params.flags |= ui_wndf_system;

        /* Make maximized windows avoid taskbar */
        params.flags |= ui_wndf_avoid;

        params.rect.p0.x = 0;
        params.rect.p0.y = 0;
        params.rect.p1.x = scr_rect.p1.x - scr_rect.p0.x;

        if (ui_is_textmode(taskbar->ui)) {
                params.rect.p1.y = 1;
                params.style &= ~ui_wds_frame;
        } else {
                params.rect.p1.y = 32;
        }

        rc = ui_window_create(taskbar->ui, &params, &taskbar->window);
        if (rc != EOK) {
                printf("Error creating window.\n");
                goto error;
        }

        rc = ui_fixed_create(&taskbar->fixed);
        if (rc != EOK) {
                printf("Error creating fixed layout.\n");
                goto error;
        }

        rc = tbsmenu_create(taskbar->window, taskbar->fixed, dspec,
            &taskbar->tbsmenu);
        if (rc != EOK) {
                printf("Error creating start menu.\n");
                goto error;
        }

        rc = tbsmenu_load(taskbar->tbsmenu, TASKBAR_CONFIG_FILE);
        if (rc != EOK) {
                printf("Error loading start menu from '%s'.\n",
                    TASKBAR_CONFIG_FILE);
        }

        rc = tbarcfg_listener_create(TBARCFG_NOTIFY_DEFAULT,
            taskbar_notif_cb, (void *)taskbar, &taskbar->lst);
        if (rc != EOK) {
                printf("Error listening for configuration changes.\n");
        }

        if (ui_is_textmode(taskbar->ui)) {
                rect.p0.x = params.rect.p0.x + 1;
                rect.p0.y = 0;
                rect.p1.x = params.rect.p0.x + 9;
                rect.p1.y = 1;
        } else {
                rect.p0.x = params.rect.p0.x + 5;
                rect.p0.y = 4;
                rect.p1.x = params.rect.p0.x + 84;
                rect.p1.y = 32 - 4;
        }

        tbsmenu_set_rect(taskbar->tbsmenu, &rect);

        rc = wndlist_create(taskbar->window, taskbar->fixed, &taskbar->wndlist);
        if (rc != EOK) {
                printf("Error creating window list.\n");
                goto error;
        }

        if (ui_is_textmode(taskbar->ui)) {
                rect.p0.x = params.rect.p0.x + 10;
                rect.p0.y = 0;
                rect.p1.x = params.rect.p1.x - 10;
                rect.p1.y = 1;
        } else {
                rect.p0.x = params.rect.p0.x + 90;
                rect.p0.y = 4;
                rect.p1.x = params.rect.p1.x - 84;
                rect.p1.y = 32 - 4;
        }
        wndlist_set_rect(taskbar->wndlist, &rect);

        /*
         * We may not be able to open WM service if display server is not
         * running. That's okay, there simply are no windows to manage.
         */
        rc = wndlist_open_wm(taskbar->wndlist, wndmgt_svc);
        if (rc != EOK && rc != ENOENT) {
                printf("Error attaching window management service.\n");
                goto error;
        }

        rc = taskbar_clock_create(taskbar->window, &taskbar->clock);
        if (rc != EOK)
                goto error;

        if (ui_is_textmode(taskbar->ui)) {
                rect.p0.x = params.rect.p1.x - 10;
                rect.p0.y = 0;
                rect.p1.x = params.rect.p1.x;
                rect.p1.y = 1;
        } else {
                rect.p0.x = params.rect.p1.x - 80;
                rect.p0.y = 4;
                rect.p1.x = params.rect.p1.x - 4;
                rect.p1.y = 32 - 4;
        }

        taskbar_clock_set_rect(taskbar->clock, &rect);

        rc = ui_fixed_add(taskbar->fixed, taskbar_clock_ctl(taskbar->clock));
        if (rc != EOK) {
                printf("Error adding control to layout.\n");
                taskbar_clock_destroy(taskbar->clock);
                goto error;
        }

        ui_window_add(taskbar->window, ui_fixed_ctl(taskbar->fixed));
        ui_window_set_cb(taskbar->window, &window_cb, (void *)taskbar);

        rc = ui_window_paint(taskbar->window);
        if (rc != EOK) {
                printf("Error painting window.\n");
                goto error;
        }

        free(dspec);
        *rtaskbar = taskbar;
        return EOK;
error:
        if (dspec != NULL)
                free(dspec);
        if (taskbar->lst != NULL)
                tbarcfg_listener_destroy(taskbar->lst);
        if (taskbar->clock != NULL)
                taskbar_clock_destroy(taskbar->clock);
        if (taskbar->wndlist != NULL)
                wndlist_destroy(taskbar->wndlist);
        if (taskbar->tbsmenu != NULL)
                tbsmenu_destroy(taskbar->tbsmenu);
        if (taskbar->window != NULL)
                ui_window_destroy(taskbar->window);
        if (taskbar->ui != NULL)
                ui_destroy(taskbar->ui);
        free(taskbar);
        return rc;

}

/** Destroy taskbar. */
void taskbar_destroy(taskbar_t *taskbar)
{
        if (taskbar->lst != NULL)
                tbarcfg_listener_destroy(taskbar->lst);
        ui_fixed_remove(taskbar->fixed, taskbar_clock_ctl(taskbar->clock));
        taskbar_clock_destroy(taskbar->clock);
        wndlist_destroy(taskbar->wndlist);
        tbsmenu_destroy(taskbar->tbsmenu);
        ui_window_destroy(taskbar->window);
        ui_destroy(taskbar->ui);
}

/** Configuration change notification callback.
 *
 * Called when configuration changed.
 *
 * @param arg Argument (taskbar_t *)
 */
static void taskbar_notif_cb(void *arg)
{
        taskbar_t *taskbar = (taskbar_t *)arg;

        ui_lock(taskbar->ui);
        tbsmenu_reload(taskbar->tbsmenu);
        ui_unlock(taskbar->ui);
}

/** @}
 */

/* [<][>][^][v][top][bottom][index][help] */
HelenOS homepage, sources at GitHub