Render window close button

This commit is contained in:
Johan Malm 2020-06-29 19:27:59 +01:00
parent 40e862f3ac
commit baca410560
17 changed files with 177 additions and 50 deletions

View file

@ -1,7 +1,7 @@
#ifndef LABWC_H #ifndef LABWC_H
#define LABWC_H #define LABWC_H
#define _POSIX_C_SOURCE 200112L #define _POSIX_C_SOURCE 200809L
#include <getopt.h> #include <getopt.h>
#include <stdbool.h> #include <stdbool.h>
#include <stdlib.h> #include <stdlib.h>
@ -93,7 +93,8 @@ struct output {
enum view_type { LAB_XDG_SHELL_VIEW, LAB_XWAYLAND_VIEW }; enum view_type { LAB_XDG_SHELL_VIEW, LAB_XWAYLAND_VIEW };
enum deco_part { enum deco_part {
LAB_DECO_PART_TITLE = 0, LAB_DECO_ICON_CLOSE = 0,
LAB_DECO_PART_TITLE,
LAB_DECO_PART_TOP, LAB_DECO_PART_TOP,
LAB_DECO_PART_RIGHT, LAB_DECO_PART_RIGHT,
LAB_DECO_PART_BOTTOM, LAB_DECO_PART_BOTTOM,

View file

@ -1,12 +1,20 @@
/*
* Theme engine for labwc - trying to be consistent with openbox
*
* Copyright Johan Malm 2020
*/
#ifndef THEME_H #ifndef THEME_H
#define THEME_H #define THEME_H
#include <stdio.h> #include <stdio.h>
#include <wlr/render/wlr_renderer.h>
struct theme { struct theme {
float window_active_title_bg_color[4]; float window_active_title_bg_color[4];
float window_active_handle_bg_color[4]; float window_active_handle_bg_color[4];
float window_inactive_title_bg_color[4]; float window_inactive_title_bg_color[4];
struct wlr_texture *xbm_close;
}; };
extern struct theme theme; extern struct theme theme;

25
include/theme/xbm/parse.h Normal file
View file

@ -0,0 +1,25 @@
/*
* Parse xbm token to create pixmap
*
* Copyright Johan Malm 2020
*/
#ifndef PARSE_H
#define PARSE_H
#include <stdint.h>
#include "theme/xbm/tokenize.h"
struct pixmap {
uint32_t *data;
int width;
int height;
};
/**
* xbm_create_pixmap - parse xbm tokens and create pixmap
* @tokens: token vector
*/
struct pixmap xbm_create_pixmap(struct token *tokens);
#endif /* PARSE_H */

View file

@ -1,7 +1,11 @@
#ifndef XBM_H /*
#define XBM_H * XBM file tokenizer
*
* Copyright Johan Malm 2020
*/
#include <cairo.h> #ifndef TOKENIZE_H
#define TOKENIZE_H
enum token_type { enum token_type {
TOKEN_NONE = 0, TOKEN_NONE = 0,
@ -18,12 +22,6 @@ struct token {
enum token_type type; enum token_type type;
}; };
/**
* xbm_create_bitmap - parse xbm tokens and create pixmap
* @tokens: token vector
*/
cairo_surface_t *xbm_create_bitmap(struct token *tokens);
/** /**
* tokenize - tokenize xbm file * tokenize - tokenize xbm file
* @buffer: buffer containing xbm file * @buffer: buffer containing xbm file
@ -38,4 +36,4 @@ struct token *xbm_tokenize(char *buffer);
*/ */
char *xbm_read_file(const char *filename); char *xbm_read_file(const char *filename);
#endif /* XBM_H */ #endif /* TOKENIZE_H */

14
include/theme/xbm/xbm.h Normal file
View file

@ -0,0 +1,14 @@
#ifndef XBM_H
#define XBM_H
#include <wlr/render/wlr_renderer.h>
#include "theme.h"
#include "theme/xbm/parse.h"
/**
* xbm_load - load theme xbm files into global theme struct
*/
void xbm_load(struct wlr_renderer *renderer);
#endif /* XBM_H */

View file

@ -42,6 +42,8 @@ wayland_protos = dependency('wayland-protocols')
xkbcommon = dependency('xkbcommon') xkbcommon = dependency('xkbcommon')
xml2 = dependency('libxml-2.0') xml2 = dependency('libxml-2.0')
glib = dependency('glib-2.0') glib = dependency('glib-2.0')
cairo = dependency('cairo')
pango = dependency('pango')
labwc_inc = include_directories('include') labwc_inc = include_directories('include')
@ -50,7 +52,7 @@ subdir('src')
subdir('tests') subdir('tests')
labwc_deps = [ labwc_deps = [
server_protos, wayland_server, wlroots, xkbcommon, xml2, glib server_protos, wayland_server, wlroots, xkbcommon, xml2, glib, cairo, pango
] ]
executable( executable(

View file

@ -1,3 +1,9 @@
/*
* Helpers for handling window decorations
*
* Copyright Johan Malm 2020
*/
#include "labwc.h" #include "labwc.h"
struct wlr_box deco_max_extents(struct view *view) struct wlr_box deco_max_extents(struct view *view)
@ -18,6 +24,12 @@ struct wlr_box deco_box(struct view *view, enum deco_part deco_part)
if (!view || !view->surface) if (!view || !view->surface)
return box; return box;
switch (deco_part) { switch (deco_part) {
case LAB_DECO_ICON_CLOSE:
box.x = view->x + view->surface->current.width - 8 - 1;
box.y = view->y - XWL_TITLEBAR_HEIGHT + 1;
box.width = 8;
box.height = 8;
break;
case LAB_DECO_PART_TITLE: case LAB_DECO_PART_TITLE:
box.x = view->x; box.x = view->x;
box.y = view->y - XWL_TITLEBAR_HEIGHT; box.y = view->y - XWL_TITLEBAR_HEIGHT;

View file

@ -1,6 +1,7 @@
#include "labwc.h" #include "labwc.h"
#include "theme.h" #include "theme.h"
#include "spawn.h" #include "spawn.h"
#include "theme/xbm/xbm.h"
struct server server = { 0 }; struct server server = { 0 };
struct rcxml rc = { 0 }; struct rcxml rc = { 0 };
@ -41,6 +42,8 @@ int main(int argc, char *argv[])
server_init(&server); server_init(&server);
server_start(&server); server_start(&server);
xbm_load(server.renderer);
if (startup_cmd) if (startup_cmd)
spawn_async_no_shell(startup_cmd); spawn_async_no_shell(startup_cmd);
wl_display_run(server.wl_display); wl_display_run(server.wl_display);

View file

@ -36,6 +36,15 @@ static void render_cycle_box(struct output *output)
} }
} }
static void render_icon(struct draw_data *d, struct wlr_box box,
struct wlr_texture *texture)
{
float matrix[9];
wlr_matrix_project_box(matrix, &box, WL_OUTPUT_TRANSFORM_NORMAL, 0,
d->transform_matrix);
wlr_render_texture_with_matrix(d->renderer, texture, matrix, 1);
}
static void render_decorations(struct wlr_output *output, struct view *view) static void render_decorations(struct wlr_output *output, struct view *view)
{ {
if (!view_want_deco(view)) if (!view_want_deco(view))
@ -56,6 +65,9 @@ static void render_decorations(struct wlr_output *output, struct view *view)
else else
ddata.rgba = theme.window_inactive_title_bg_color; ddata.rgba = theme.window_inactive_title_bg_color;
draw_rect(&ddata, deco_box(view, LAB_DECO_PART_TITLE)); draw_rect(&ddata, deco_box(view, LAB_DECO_PART_TITLE));
render_icon(&ddata, deco_box(view, LAB_DECO_ICON_CLOSE),
theme.xbm_close);
} }
struct render_data { struct render_data {

View file

@ -1,3 +1,5 @@
labwc_sources += files( labwc_sources += files(
'theme.c', 'theme.c',
) )
subdir('xbm')

View file

@ -0,0 +1,5 @@
labwc_sources += files(
'parse.c',
'tokenize.c',
'xbm.c',
)

View file

@ -1,3 +1,9 @@
/*
* Parse xbm token to create pixmap
*
* Copyright Johan Malm 2020
*/
#define _POSIX_C_SOURCE 200809L #define _POSIX_C_SOURCE 200809L
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
@ -6,17 +12,17 @@
#include <stdbool.h> #include <stdbool.h>
#include "buf.h" #include "buf.h"
#include "xbm.h" #include "theme/xbm/parse.h"
static unsigned char defaultcolor[] = { 127, 127, 127, 255 }; /* TODO: should be window.active.button.unpressed.image.color */
static unsigned char background[] = { 255, 255, 255, 255 }; static unsigned char defaultcolor[] = { 255, 255, 255, 255 };
static uint32_t *pixmap; static uint32_t *data;
static void add_pixel(int position, unsigned char *rbga) static void add_pixel(int position, unsigned char *rbga)
{ {
pixmap[position] = (rbga[3] << 24) | (rbga[0] << 16) | (rbga[1] << 8) | data[position] = (rbga[3] << 24) | (rbga[0] << 16) | (rbga[1] << 8) |
rbga[0]; rbga[0];
} }
static void init_pixmap(int w, int h) static void init_pixmap(int w, int h)
@ -25,7 +31,7 @@ static void init_pixmap(int w, int h)
if (has_run) if (has_run)
return; return;
has_run = true; has_run = true;
pixmap = (uint32_t *)calloc(w * h, sizeof(uint32_t)); data = (uint32_t *)calloc(w * h, sizeof(uint32_t));
} }
static void process_bytes(int height, int width, struct token *tokens) static void process_bytes(int height, int width, struct token *tokens)
@ -43,22 +49,16 @@ static void process_bytes(int height, int width, struct token *tokens)
return; return;
int value = (int)strtol(t->name, NULL, 0); int value = (int)strtol(t->name, NULL, 0);
int bit = 1 << (col % 8); int bit = 1 << (col % 8);
if (value & bit) { if (value & bit)
add_pixel(row * width + col, defaultcolor); add_pixel(row * width + col, defaultcolor);
printf(".");
} else {
add_pixel(row * width + col, background);
printf(" ");
}
} }
++t; ++t;
printf("\n");
} }
} }
cairo_surface_t *xbm_create_bitmap(struct token *tokens) struct pixmap xbm_create_pixmap(struct token *tokens)
{ {
cairo_surface_t *g_surface; struct pixmap pixmap;
int width = 0, height = 0; int width = 0, height = 0;
for (struct token *t = tokens; t->type; t++) { for (struct token *t = tokens; t->type; t++) {
if (width && height) { if (width && height) {
@ -72,20 +72,11 @@ cairo_surface_t *xbm_create_bitmap(struct token *tokens)
else if (strstr(t->name, "height")) else if (strstr(t->name, "height"))
height = atoi((++t)->name); height = atoi((++t)->name);
} }
out: out:
g_surface = pixmap.data = data;
cairo_image_surface_create(CAIRO_FORMAT_ARGB32, width, height); pixmap.width = width;
if (!g_surface) { pixmap.height = height;
fprintf(stderr, "no surface\n"); return pixmap;
return NULL;
}
unsigned char *surface_data = cairo_image_surface_get_data(g_surface);
cairo_surface_flush(g_surface);
memcpy(surface_data, pixmap, width * height * 4);
free(pixmap);
cairo_surface_mark_dirty(g_surface);
return g_surface;
} }
char *xbm_read_file(const char *filename) char *xbm_read_file(const char *filename)

View file

@ -1,8 +1,14 @@
/*
* XBM file tokenizer
*
* Copyright Johan Malm 2020
*/
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include "xbm.h" #include "theme/xbm/tokenize.h"
static char *current_buffer_position; static char *current_buffer_position;
static struct token *tokens; static struct token *tokens;

34
src/theme/xbm/xbm.c Normal file
View file

@ -0,0 +1,34 @@
/*
* Create wlr textures based on xbm data
*
* Copyright Johan Malm 2020
*/
#include <stdio.h>
#include <stdlib.h>
#include "theme/xbm/xbm.h"
#include "theme/xbm/parse.h"
static char filename[] = "/usr/share/themes/Bear2/openbox-3/close.xbm";
void xbm_load(struct wlr_renderer *renderer)
{
struct token *tokens;
char *buffer = xbm_read_file(filename);
if (!buffer) {
fprintf(stderr, "no buffer\n");
return;
}
tokens = xbm_tokenize(buffer);
free(buffer);
struct pixmap pixmap = xbm_create_pixmap(tokens);
free(tokens);
theme.xbm_close = wlr_texture_from_pixels(
renderer, WL_SHM_FORMAT_ARGB8888, pixmap.width * 4,
pixmap.width, pixmap.height, pixmap.data);
if (pixmap.data)
free(pixmap.data);
}

View file

@ -1,2 +1,3 @@
xbm-tokenize xbm-tokenize
xbm-parse xbm-parse
*.png

View file

@ -1,7 +1,9 @@
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h>
#include <cairo.h>
#include "xbm.h" #include "theme/xbm/parse.h"
int main(int argc, char **argv) int main(int argc, char **argv)
{ {
@ -17,18 +19,29 @@ int main(int argc, char **argv)
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
tokens = xbm_tokenize(buffer); tokens = xbm_tokenize(buffer);
free(buffer); free(buffer);
struct pixmap pixmap = xbm_create_pixmap(tokens);
cairo_surface_t *surface = xbm_create_bitmap(tokens);
free(tokens); free(tokens);
if (!surface) cairo_surface_t *g_surface;
g_surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32,
pixmap.width, pixmap.height);
if (!g_surface) {
fprintf(stderr, "no surface\n"); fprintf(stderr, "no surface\n");
exit(EXIT_FAILURE);
}
unsigned char *surface_data = cairo_image_surface_get_data(g_surface);
cairo_surface_flush(g_surface);
memcpy(surface_data, pixmap.data, pixmap.width * pixmap.height * 4);
if (pixmap.data)
free(pixmap.data);
cairo_surface_mark_dirty(g_surface);
char png_name[1024]; char png_name[1024];
snprintf(png_name, sizeof(png_name), "%s.png", argv[1]); snprintf(png_name, sizeof(png_name), "%s.png", argv[1]);
if (cairo_surface_write_to_png(surface, png_name)) { if (cairo_surface_write_to_png(g_surface, png_name)) {
fprintf(stderr, "cannot save png\n"); fprintf(stderr, "cannot save png\n");
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
cairo_surface_destroy(surface); cairo_surface_destroy(g_surface);
exit(EXIT_SUCCESS); exit(EXIT_SUCCESS);
} }

View file

@ -4,7 +4,7 @@
#include <string.h> #include <string.h>
#include "buf.h" #include "buf.h"
#include "xbm.h" #include "theme/xbm/tokenize.h"
/* Read file into buffer, because it's easier to tokenize that way */ /* Read file into buffer, because it's easier to tokenize that way */
char *read_file(const char *filename) char *read_file(const char *filename)