2020-10-19 22:14:17 +01:00
|
|
|
#define _POSIX_C_SOURCE 200809L
|
2021-02-17 20:38:16 +00:00
|
|
|
#include <assert.h>
|
2021-06-26 18:28:27 +01:00
|
|
|
#include <cairo.h>
|
2021-02-17 20:38:16 +00:00
|
|
|
#include <ctype.h>
|
2021-03-13 21:08:14 +00:00
|
|
|
#include <drm_fourcc.h>
|
2021-02-17 20:38:16 +00:00
|
|
|
#include <libxml/parser.h>
|
|
|
|
|
#include <libxml/tree.h>
|
2020-10-19 22:14:17 +01:00
|
|
|
#include <pango/pangocairo.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
2021-02-17 20:38:16 +00:00
|
|
|
#include <strings.h>
|
|
|
|
|
#include "common/buf.h"
|
|
|
|
|
#include "common/dir.h"
|
2020-10-21 20:32:08 +01:00
|
|
|
#include "common/font.h"
|
2021-02-17 20:38:16 +00:00
|
|
|
#include "common/nodename.h"
|
|
|
|
|
#include "common/string-helpers.h"
|
2020-10-19 22:14:17 +01:00
|
|
|
#include "labwc.h"
|
|
|
|
|
#include "menu/menu.h"
|
2021-02-21 22:18:34 +00:00
|
|
|
#include "theme.h"
|
2020-10-19 22:14:17 +01:00
|
|
|
|
2021-02-16 20:43:20 +00:00
|
|
|
static const char font[] = "Sans 8";
|
2020-10-21 20:32:08 +01:00
|
|
|
|
2021-02-17 20:38:16 +00:00
|
|
|
/* state-machine variables for processing <item></item> */
|
|
|
|
|
static bool in_item = false;
|
|
|
|
|
static struct menuitem *current_item;
|
|
|
|
|
|
|
|
|
|
#define MENUWIDTH (120)
|
2020-10-21 20:32:08 +01:00
|
|
|
#define MENUHEIGHT (25)
|
|
|
|
|
#define MENU_PADDING_WIDTH (7)
|
2020-10-19 22:14:17 +01:00
|
|
|
|
|
|
|
|
struct wlr_texture *
|
|
|
|
|
texture_create(struct server *server, struct wlr_box *geo, const char *text,
|
|
|
|
|
float *bg, float *fg)
|
|
|
|
|
{
|
|
|
|
|
cairo_surface_t *surf = cairo_image_surface_create(CAIRO_FORMAT_ARGB32,
|
|
|
|
|
geo->width, geo->height);
|
|
|
|
|
cairo_t *cairo = cairo_create(surf);
|
|
|
|
|
|
|
|
|
|
cairo_set_source_rgb(cairo, bg[0], bg[1], bg[2]);
|
|
|
|
|
cairo_paint(cairo);
|
|
|
|
|
cairo_set_source_rgba(cairo, fg[0], fg[1], fg[2], fg[3]);
|
|
|
|
|
cairo_move_to(cairo, 0, 0);
|
|
|
|
|
|
|
|
|
|
PangoLayout *layout = pango_cairo_create_layout(cairo);
|
|
|
|
|
pango_layout_set_width(layout, geo->width * PANGO_SCALE);
|
|
|
|
|
pango_layout_set_text(layout, text, -1);
|
|
|
|
|
|
|
|
|
|
PangoFontDescription *desc = pango_font_description_from_string(font);
|
|
|
|
|
pango_layout_set_font_description(layout, desc);
|
|
|
|
|
pango_font_description_free(desc);
|
|
|
|
|
pango_cairo_update_layout(cairo, layout);
|
2020-10-21 20:32:08 +01:00
|
|
|
|
|
|
|
|
/* center-align vertically */
|
|
|
|
|
int height;
|
|
|
|
|
pango_layout_get_pixel_size(layout, NULL, &height);
|
|
|
|
|
cairo_move_to(cairo, MENU_PADDING_WIDTH, (geo->height - height) / 2);
|
|
|
|
|
|
2020-10-19 22:14:17 +01:00
|
|
|
pango_cairo_show_layout(cairo, layout);
|
|
|
|
|
g_object_unref(layout);
|
|
|
|
|
|
|
|
|
|
cairo_surface_flush(surf);
|
|
|
|
|
unsigned char *data = cairo_image_surface_get_data(surf);
|
|
|
|
|
struct wlr_texture *texture = wlr_texture_from_pixels(server->renderer,
|
2021-03-13 21:08:14 +00:00
|
|
|
DRM_FORMAT_ARGB8888, cairo_image_surface_get_stride(surf),
|
2020-10-19 22:14:17 +01:00
|
|
|
geo->width, geo->height, data);
|
|
|
|
|
|
|
|
|
|
cairo_destroy(cairo);
|
|
|
|
|
cairo_surface_destroy(surf);
|
|
|
|
|
return texture;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct menuitem *
|
2021-02-17 20:38:16 +00:00
|
|
|
menuitem_create(struct server *server, struct menu *menu, const char *text)
|
2020-10-19 22:14:17 +01:00
|
|
|
{
|
|
|
|
|
struct menuitem *menuitem = calloc(1, sizeof(struct menuitem));
|
|
|
|
|
if (!menuitem) {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2021-02-21 21:54:40 +00:00
|
|
|
struct theme *theme = server->theme;
|
2020-10-19 22:14:17 +01:00
|
|
|
menuitem->geo_box.width = MENUWIDTH;
|
|
|
|
|
menuitem->geo_box.height = MENUHEIGHT;
|
|
|
|
|
menuitem->active_texture = texture_create(server, &menuitem->geo_box,
|
2021-02-21 21:54:40 +00:00
|
|
|
text, theme->menu_items_active_bg_color,
|
|
|
|
|
theme->menu_items_active_text_color);
|
2020-10-19 22:14:17 +01:00
|
|
|
menuitem->inactive_texture = texture_create(server, &menuitem->geo_box,
|
2021-02-21 21:54:40 +00:00
|
|
|
text, theme->menu_items_bg_color, theme->menu_items_text_color);
|
2020-10-19 22:14:17 +01:00
|
|
|
wl_list_insert(&menu->menuitems, &menuitem->link);
|
|
|
|
|
return menuitem;
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-19 23:31:14 +00:00
|
|
|
static void fill_item(char *nodename, char *content, struct menu *menu)
|
2021-02-17 20:38:16 +00:00
|
|
|
{
|
|
|
|
|
string_truncate_at_pattern(nodename, ".item.menu");
|
|
|
|
|
|
|
|
|
|
if (getenv("LABWC_DEBUG_MENU_NODENAMES")) {
|
|
|
|
|
printf("%s: %s\n", nodename, content);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Handle the following:
|
|
|
|
|
* <item label="">
|
|
|
|
|
* <action name="">
|
|
|
|
|
* <command></command>
|
|
|
|
|
* </action>
|
|
|
|
|
* </item>
|
|
|
|
|
*/
|
|
|
|
|
if (!strcmp(nodename, "label")) {
|
2021-02-19 23:31:14 +00:00
|
|
|
current_item = menuitem_create(menu->server, menu, content);
|
2021-02-17 20:38:16 +00:00
|
|
|
}
|
|
|
|
|
assert(current_item);
|
|
|
|
|
if (!strcmp(nodename, "name.action")) {
|
|
|
|
|
current_item->action = strdup(content);
|
|
|
|
|
} else if (!strcmp(nodename, "command.action")) {
|
|
|
|
|
current_item->command = strdup(content);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2021-02-19 23:31:14 +00:00
|
|
|
entry(xmlNode *node, char *nodename, char *content, struct menu *menu)
|
2021-02-17 20:38:16 +00:00
|
|
|
{
|
|
|
|
|
static bool in_root_menu = false;
|
|
|
|
|
|
|
|
|
|
if (!nodename) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
string_truncate_at_pattern(nodename, ".openbox_menu");
|
|
|
|
|
if (!content) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!strcmp(nodename, "id.menu")) {
|
|
|
|
|
in_root_menu = !strcmp(content, "root-menu") ? true : false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* We only handle the root-menu for the time being */
|
|
|
|
|
if (!in_root_menu) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (in_item) {
|
2021-02-19 23:31:14 +00:00
|
|
|
fill_item(nodename, content, menu);
|
2021-02-17 20:38:16 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2021-02-19 23:31:14 +00:00
|
|
|
process_node(xmlNode *node, struct menu *menu)
|
2021-02-17 20:38:16 +00:00
|
|
|
{
|
|
|
|
|
char *content;
|
|
|
|
|
static char buffer[256];
|
|
|
|
|
char *name;
|
|
|
|
|
|
|
|
|
|
content = (char *)node->content;
|
|
|
|
|
if (xmlIsBlankNode(node)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
name = nodename(node, buffer, sizeof(buffer));
|
2021-02-19 23:31:14 +00:00
|
|
|
entry(node, name, content, menu);
|
2021-02-17 20:38:16 +00:00
|
|
|
}
|
|
|
|
|
|
2021-02-19 23:31:14 +00:00
|
|
|
static void xml_tree_walk(xmlNode *node, struct menu *menu);
|
2021-02-17 20:38:16 +00:00
|
|
|
|
|
|
|
|
static void
|
2021-02-19 23:31:14 +00:00
|
|
|
traverse(xmlNode *n, struct menu *menu)
|
2021-02-17 20:38:16 +00:00
|
|
|
{
|
2021-02-19 23:31:14 +00:00
|
|
|
process_node(n, menu);
|
2021-02-17 20:38:16 +00:00
|
|
|
for (xmlAttr *attr = n->properties; attr; attr = attr->next) {
|
2021-02-19 23:31:14 +00:00
|
|
|
xml_tree_walk(attr->children, menu);
|
2021-02-17 20:38:16 +00:00
|
|
|
}
|
2021-02-19 23:31:14 +00:00
|
|
|
xml_tree_walk(n->children, menu);
|
2021-02-17 20:38:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2021-02-19 23:31:14 +00:00
|
|
|
xml_tree_walk(xmlNode *node, struct menu *menu)
|
2021-02-17 20:38:16 +00:00
|
|
|
{
|
|
|
|
|
for (xmlNode *n = node; n && n->name; n = n->next) {
|
|
|
|
|
if (!strcasecmp((char *)n->name, "comment")) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (!strcasecmp((char *)n->name, "item")) {
|
|
|
|
|
in_item = true;
|
2021-02-19 23:31:14 +00:00
|
|
|
traverse(n, menu);
|
2021-02-17 20:38:16 +00:00
|
|
|
in_item = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2021-02-19 23:31:14 +00:00
|
|
|
traverse(n, menu);
|
2021-02-17 20:38:16 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2021-02-19 23:31:14 +00:00
|
|
|
parse_xml(const char *filename, struct menu *menu)
|
2021-02-17 20:38:16 +00:00
|
|
|
{
|
|
|
|
|
FILE *stream;
|
|
|
|
|
char *line = NULL;
|
|
|
|
|
size_t len = 0;
|
|
|
|
|
struct buf b;
|
|
|
|
|
static char menuxml[4096] = { 0 };
|
|
|
|
|
|
|
|
|
|
if (!strlen(config_dir())) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-02-19 23:05:14 +00:00
|
|
|
snprintf(menuxml, sizeof(menuxml), "%s/%s", config_dir(), filename);
|
2021-02-17 20:38:16 +00:00
|
|
|
|
|
|
|
|
stream = fopen(menuxml, "r");
|
|
|
|
|
if (!stream) {
|
|
|
|
|
warn("cannot read (%s)", menuxml);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
info("read menu file (%s)", menuxml);
|
|
|
|
|
buf_init(&b);
|
|
|
|
|
while (getline(&line, &len, stream) != -1) {
|
|
|
|
|
char *p = strrchr(line, '\n');
|
|
|
|
|
if (p)
|
|
|
|
|
*p = '\0';
|
|
|
|
|
buf_add(&b, line);
|
|
|
|
|
}
|
|
|
|
|
free(line);
|
|
|
|
|
fclose(stream);
|
2021-02-19 23:05:14 +00:00
|
|
|
xmlDoc *d = xmlParseMemory(b.buf, b.len);
|
|
|
|
|
if (!d) {
|
|
|
|
|
warn("xmlParseMemory()");
|
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
|
}
|
2021-02-19 23:31:14 +00:00
|
|
|
xml_tree_walk(xmlDocGetRootElement(d), menu);
|
2021-02-19 23:05:14 +00:00
|
|
|
xmlFreeDoc(d);
|
|
|
|
|
xmlCleanupParser();
|
2021-02-17 20:38:16 +00:00
|
|
|
free(b.buf);
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-19 22:14:17 +01:00
|
|
|
void
|
2021-02-19 23:05:14 +00:00
|
|
|
menu_init_rootmenu(struct server *server, struct menu *menu)
|
2020-10-19 22:14:17 +01:00
|
|
|
{
|
2021-02-17 20:38:16 +00:00
|
|
|
static bool has_run;
|
|
|
|
|
|
2021-02-19 23:05:14 +00:00
|
|
|
if (!has_run) {
|
|
|
|
|
LIBXML_TEST_VERSION
|
|
|
|
|
wl_list_init(&menu->menuitems);
|
|
|
|
|
server->rootmenu = menu;
|
2021-02-19 23:31:14 +00:00
|
|
|
menu->server = server;
|
2021-02-17 20:38:16 +00:00
|
|
|
}
|
|
|
|
|
|
2021-02-19 23:31:14 +00:00
|
|
|
parse_xml("menu.xml", menu);
|
2021-02-19 23:05:14 +00:00
|
|
|
|
2021-02-17 20:38:16 +00:00
|
|
|
/* Default menu if no menu.xml found */
|
2021-02-19 23:05:14 +00:00
|
|
|
if (wl_list_empty(&menu->menuitems)) {
|
2021-02-17 20:38:16 +00:00
|
|
|
current_item = menuitem_create(server, menu, "Reconfigure");
|
|
|
|
|
current_item->action = strdup("Reconfigure");
|
|
|
|
|
current_item = menuitem_create(server, menu, "Exit");
|
|
|
|
|
current_item->action = strdup("Exit");
|
|
|
|
|
}
|
2020-10-19 22:14:17 +01:00
|
|
|
menu_move(menu, 100, 100);
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-22 19:43:27 +01:00
|
|
|
void
|
|
|
|
|
menu_finish(struct menu *menu)
|
|
|
|
|
{
|
|
|
|
|
struct menuitem *menuitem, *next;
|
|
|
|
|
wl_list_for_each_safe(menuitem, next, &menu->menuitems, link) {
|
|
|
|
|
if (menuitem->action)
|
|
|
|
|
free(menuitem->action);
|
|
|
|
|
if (menuitem->command)
|
|
|
|
|
free(menuitem->command);
|
|
|
|
|
if (menuitem->active_texture)
|
|
|
|
|
free(menuitem->active_texture);
|
|
|
|
|
if (menuitem->inactive_texture)
|
|
|
|
|
free(menuitem->inactive_texture);
|
|
|
|
|
wl_list_remove(&menuitem->link);
|
|
|
|
|
free(menuitem);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-19 22:14:17 +01:00
|
|
|
void
|
|
|
|
|
menu_move(struct menu *menu, int x, int y)
|
|
|
|
|
{
|
|
|
|
|
menu->x = x;
|
|
|
|
|
menu->y = y;
|
|
|
|
|
|
|
|
|
|
int offset = 0;
|
|
|
|
|
struct menuitem *menuitem;
|
2020-10-21 20:32:08 +01:00
|
|
|
wl_list_for_each_reverse (menuitem, &menu->menuitems, link) {
|
2020-10-19 22:14:17 +01:00
|
|
|
menuitem->geo_box.x = menu->x;
|
|
|
|
|
menuitem->geo_box.y = menu->y + offset;
|
|
|
|
|
offset += menuitem->geo_box.height;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
menu_set_selected(struct menu *menu, int x, int y)
|
|
|
|
|
{
|
|
|
|
|
struct menuitem *menuitem;
|
|
|
|
|
wl_list_for_each (menuitem, &menu->menuitems, link) {
|
|
|
|
|
menuitem->selected =
|
|
|
|
|
wlr_box_contains_point(&menuitem->geo_box, x, y);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
menu_action_selected(struct server *server, struct menu *menu)
|
|
|
|
|
{
|
|
|
|
|
struct menuitem *menuitem;
|
|
|
|
|
wl_list_for_each (menuitem, &menu->menuitems, link) {
|
|
|
|
|
if (menuitem->selected) {
|
|
|
|
|
action(server, menuitem->action, menuitem->command);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-02-19 23:05:14 +00:00
|
|
|
|
|
|
|
|
void
|
2021-02-19 23:31:14 +00:00
|
|
|
menu_reconfigure(struct server *server, struct menu *menu)
|
2021-02-19 23:05:14 +00:00
|
|
|
{
|
2021-02-19 23:31:14 +00:00
|
|
|
menu_finish(menu);
|
|
|
|
|
menu_init_rootmenu(server, menu);
|
2021-02-19 23:05:14 +00:00
|
|
|
}
|