Refactor draw.c

This is a big patch that does not change any behaviour. We abstract the data
needed by the draw.c functions into a context object, and pass that around
instead of passing each argument every time. This results in a much cleaner API
- most functions lose 3-4 arguments.

This is a preparatory patch for further work on statusbar.
This commit is contained in:
Aldo Cortesi 2007-12-15 10:12:17 +11:00 committed by Julien Danjou
parent 7327a92972
commit f36a56cdcc
6 changed files with 129 additions and 121 deletions

View File

@ -228,7 +228,7 @@ setup_screen(awesome_config *awesomeconf, int screen)
setup(awesomeconf, screen); setup(awesomeconf, screen);
statusbar_init(awesomeconf->display, screen, &awesomeconf->screens[screen].statusbar, statusbar_init(awesomeconf->display, screen, &awesomeconf->screens[screen].statusbar,
awesomeconf->cursor[CurNormal], awesomeconf->screens[screen].font, awesomeconf->cursor[CurNormal], awesomeconf->screens[screen].font,
awesomeconf->screens[screen].layouts, &awesomeconf->screens[screen].padding); &awesomeconf->screens[screen].padding);
} }
/** Startup Error handler to check if another window manager /** Startup Error handler to check if another window manager

74
draw.c
View File

@ -27,8 +27,33 @@
#include "util.h" #include "util.h"
#include "draw.h" #include "draw.h"
DrawCtx *
draw_get_context(Display *display, int phys_screen, int width, int height)
{
DrawCtx *d;
d = p_new(DrawCtx, 1);
d->phys_screen = phys_screen;
d->display = display;
d->width = width;
d->height = height;
d->depth = DefaultDepth(display, phys_screen);
d->visual = DefaultVisual(display, phys_screen);
d->drawable = XCreatePixmap(display,
RootWindow(display, phys_screen),
width, height, d->depth);
return d;
};
void
draw_free_context(DrawCtx *ctx)
{
XFreePixmap(ctx->display, ctx->drawable);
p_delete(&ctx);
}
void void
drawtext(Display *disp, int screen, int x, int y, int w, int h, Drawable drawable, int dw, int dh, XftFont *font, const char *text, XColor color[ColLast]) drawtext(DrawCtx *ctx, int x, int y, int w, int h, XftFont *font, const char *text, XColor color[ColLast])
{ {
int nw = 0; int nw = 0;
static char buf[256]; static char buf[256];
@ -37,11 +62,11 @@ drawtext(Display *disp, int screen, int x, int y, int w, int h, Drawable drawabl
cairo_surface_t *surface; cairo_surface_t *surface;
cairo_t *cr; cairo_t *cr;
drawrectangle(disp, screen, x, y, w, h, drawable, dw, dh, True, color[ColBG]); drawrectangle(ctx, x, y, w, h, True, color[ColBG]);
if(!a_strlen(text)) if(!a_strlen(text))
return; return;
surface = cairo_xlib_surface_create(disp, drawable, DefaultVisual(disp, screen), dw, dh); surface = cairo_xlib_surface_create(ctx->display, ctx->drawable, ctx->visual, ctx->width, ctx->height);
cr = cairo_create(surface); cr = cairo_create(surface);
font_face = cairo_ft_font_face_create_for_pattern(font->pattern); font_face = cairo_ft_font_face_create_for_pattern(font->pattern);
cairo_set_font_face(cr, font_face); cairo_set_font_face(cr, font_face);
@ -53,7 +78,7 @@ drawtext(Display *disp, int screen, int x, int y, int w, int h, Drawable drawabl
len = sizeof(buf) - 1; len = sizeof(buf) - 1;
memcpy(buf, text, len); memcpy(buf, text, len);
buf[len] = 0; buf[len] = 0;
while(len && (nw = textwidth(disp, font, buf)) > w) while(len && (nw = textwidth(ctx, font, buf)) > w)
buf[--len] = 0; buf[--len] = 0;
if(nw > w) if(nw > w)
return; /* too long */ return; /* too long */
@ -67,7 +92,7 @@ drawtext(Display *disp, int screen, int x, int y, int w, int h, Drawable drawabl
buf[len - 3] = '.'; buf[len - 3] = '.';
} }
cairo_move_to(cr, x + font->height / 2, y + font->ascent + (dh - font->height) / 2); cairo_move_to(cr, x + font->height / 2, y + font->ascent + (ctx->height - font->height) / 2);
cairo_show_text(cr, buf); cairo_show_text(cr, buf);
cairo_font_face_destroy(font_face); cairo_font_face_destroy(font_face);
@ -76,12 +101,12 @@ drawtext(Display *disp, int screen, int x, int y, int w, int h, Drawable drawabl
} }
void void
drawrectangle(Display *disp, int screen, int x, int y, int w, int h, Drawable drawable, int dw, int dh, Bool filled, XColor color) drawrectangle(DrawCtx *ctx, int x, int y, int w, int h, Bool filled, XColor color)
{ {
cairo_surface_t *surface; cairo_surface_t *surface;
cairo_t *cr; cairo_t *cr;
surface = cairo_xlib_surface_create(disp, drawable, DefaultVisual(disp, screen), dw, dh); surface = cairo_xlib_surface_create(ctx->display, ctx->drawable, ctx->visual, ctx->width, ctx->height);
cr = cairo_create (surface); cr = cairo_create (surface);
cairo_set_antialias(cr, CAIRO_ANTIALIAS_NONE); cairo_set_antialias(cr, CAIRO_ANTIALIAS_NONE);
@ -102,12 +127,12 @@ drawrectangle(Display *disp, int screen, int x, int y, int w, int h, Drawable dr
} }
void void
drawcircle(Display *disp, int screen, int x, int y, int r, Drawable drawable, int dw, int dh, Bool filled, XColor color) drawcircle(DrawCtx *ctx, int x, int y, int r, Bool filled, XColor color)
{ {
cairo_surface_t *surface; cairo_surface_t *surface;
cairo_t *cr; cairo_t *cr;
surface = cairo_xlib_surface_create(disp, drawable, DefaultVisual(disp, screen), dw, dh); surface = cairo_xlib_surface_create(ctx->display, ctx->drawable, ctx->visual, ctx->width, ctx->height);
cr = cairo_create (surface); cr = cairo_create (surface);
cairo_set_line_width(cr, 1.0); cairo_set_line_width(cr, 1.0);
cairo_set_source_rgb(cr, color.red / 65535.0, color.green / 65535.0, color.blue / 65535.0); cairo_set_source_rgb(cr, color.red / 65535.0, color.green / 65535.0, color.blue / 65535.0);
@ -126,18 +151,18 @@ drawcircle(Display *disp, int screen, int x, int y, int r, Drawable drawable, in
} }
Drawable Drawable
draw_rotate(Display *disp, int screen, Drawable drawable, int dw, int dh, double angle, int tx, int ty) draw_rotate(DrawCtx *ctx, int screen, double angle, int tx, int ty)
{ {
cairo_surface_t *surface, *source; cairo_surface_t *surface, *source;
cairo_t *cr; cairo_t *cr;
Drawable newdrawable; Drawable newdrawable;
newdrawable = XCreatePixmap(disp, newdrawable = XCreatePixmap(ctx->display,
RootWindow(disp, screen), RootWindow(ctx->display, screen),
dh, dw, ctx->height, ctx->width,
DefaultDepth(disp, screen)); ctx->depth);
surface = cairo_xlib_surface_create(disp, newdrawable, DefaultVisual(disp, screen), dh, dw); surface = cairo_xlib_surface_create(ctx->display, newdrawable, ctx->visual, ctx->height, ctx->width);
source = cairo_xlib_surface_create(disp, drawable, DefaultVisual(disp, screen), dw, dh); source = cairo_xlib_surface_create(ctx->display, ctx->drawable, ctx->visual, ctx->width, ctx->height);
cr = cairo_create (surface); cr = cairo_create (surface);
cairo_translate(cr, tx, ty); cairo_translate(cr, tx, ty);
@ -153,18 +178,19 @@ draw_rotate(Display *disp, int screen, Drawable drawable, int dw, int dh, double
return newdrawable; return newdrawable;
} }
unsigned short unsigned short
textwidth(Display *disp, XftFont *font, char *text) textwidth_primitive(Display *display, XftFont *font, char *text)
{ {
cairo_surface_t *surface; cairo_surface_t *surface;
cairo_t *cr; cairo_t *cr;
cairo_font_face_t *font_face; cairo_font_face_t *font_face;
cairo_text_extents_t te; cairo_text_extents_t te;
surface = cairo_xlib_surface_create(disp, DefaultScreen(disp), surface = cairo_xlib_surface_create(display, DefaultScreen(display),
DefaultVisual(disp, DefaultScreen(disp)), DefaultVisual(display, DefaultScreen(display)),
DisplayWidth(disp, DefaultScreen(disp)), DisplayWidth(display, DefaultScreen(display)),
DisplayHeight(disp, DefaultScreen(disp))); DisplayHeight(display, DefaultScreen(display)));
cr = cairo_create(surface); cr = cairo_create(surface);
font_face = cairo_ft_font_face_create_for_pattern(font->pattern); font_face = cairo_ft_font_face_create_for_pattern(font->pattern);
cairo_set_font_face(cr, font_face); cairo_set_font_face(cr, font_face);
@ -177,4 +203,10 @@ textwidth(Display *disp, XftFont *font, char *text)
return MAX(te.x_advance, te.width) + font->height; return MAX(te.x_advance, te.width) + font->height;
} }
unsigned short
textwidth(DrawCtx *ctx, XftFont *font, char *text)
{
return textwidth_primitive(ctx->display, font, text);
}
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99 // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99

25
draw.h
View File

@ -24,10 +24,25 @@
#include "config.h" #include "config.h"
void drawtext(Display *, int, int, int, int, int, Drawable, int, int, XftFont *, const char *, XColor []); typedef struct DrawCtx DrawCtx;
void drawrectangle(Display *, int, int, int, int, int, Drawable, int, int, Bool, XColor); struct DrawCtx
void drawcircle(Display *, int, int, int, int, Drawable, int, int, Bool, XColor); {
Drawable draw_rotate(Display *, int, Drawable, int, int, double, int, int); Display *display;
unsigned short textwidth(Display *, XftFont *, char *); Drawable drawable;
Visual *visual;
int width;
int height;
int phys_screen;
int depth;
};
DrawCtx *draw_get_context(Display*, int, int, int);
void draw_free_context(DrawCtx*);
void drawtext(DrawCtx *, int, int, int, int, XftFont *, const char *, XColor []);
void drawrectangle(DrawCtx *, int, int, int, int, Bool, XColor);
void drawcircle(DrawCtx *, int, int, int, Bool, XColor);
Drawable draw_rotate(DrawCtx *, int, double, int, int);
unsigned short textwidth(DrawCtx *, XftFont *, char *);
unsigned short textwidth_primitive(Display*, XftFont*, char*);
#endif #endif
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99 // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99

View File

@ -73,7 +73,7 @@ handle_event_buttonpress(XEvent * e, awesome_config *awesomeconf)
{ {
for(i = 1, tag = awesomeconf->screens[screen].tags; tag; tag = tag->next, i++) for(i = 1, tag = awesomeconf->screens[screen].tags; tag; tag = tag->next, i++)
{ {
x += textwidth(e->xany.display, awesomeconf->screens[screen].font, tag->name); x += textwidth_primitive(e->xany.display, awesomeconf->screens[screen].font, awesomeconf->screens[screen].tags[i].name);
if(((awesomeconf->screens[screen].statusbar.position == BarTop if(((awesomeconf->screens[screen].statusbar.position == BarTop
|| awesomeconf->screens[screen].statusbar.position == BarBot) || awesomeconf->screens[screen].statusbar.position == BarBot)
&& ev->x < x) && ev->x < x)

View File

@ -52,157 +52,122 @@ statusbar_draw(awesome_config *awesomeconf, int screen)
{ {
int z, x = 0, y = 0, w; int z, x = 0, y = 0, w;
Client *sel = awesomeconf->focus->client; Client *sel = awesomeconf->focus->client;
Drawable drawable;
int phys_screen = get_phys_screen(awesomeconf->display, screen); int phys_screen = get_phys_screen(awesomeconf->display, screen);
Statusbar sbar;
Tag *tag; Tag *tag;
Layout *layout;
sbar = awesomeconf->screens[screen].statusbar;
DrawCtx *ctx = draw_get_context(awesomeconf->display, phys_screen, sbar.width, sbar.height);
/* don't waste our time */ /* don't waste our time */
if(awesomeconf->screens[screen].statusbar.position == BarOff) if(sbar.position == BarOff)
return; return;
drawable = XCreatePixmap(awesomeconf->display,
RootWindow(awesomeconf->display, phys_screen),
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
DefaultDepth(awesomeconf->display, phys_screen));
for(tag = awesomeconf->screens[screen].tags; tag; tag = tag->next) for(tag = awesomeconf->screens[screen].tags; tag; tag = tag->next)
{ {
w = textwidth(awesomeconf->display, awesomeconf->screens[screen].font, tag->name); w = textwidth(ctx, awesomeconf->screens[screen].font, tag->name);
if(tag->selected) if(tag->selected)
{ {
drawtext(awesomeconf->display, phys_screen, drawtext(ctx, x, y, w,
x, y, w, sbar.height,
awesomeconf->screens[screen].statusbar.height,
drawable,
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
awesomeconf->screens[screen].font, awesomeconf->screens[screen].font,
tag->name, awesomeconf->screens[screen].colors_selected); tag->name,
awesomeconf->screens[screen].colors_selected);
if(isoccupied(awesomeconf->screens[screen].tclink, screen, awesomeconf->clients, tag)) if(isoccupied(awesomeconf->screens[screen].tclink, screen, awesomeconf->clients, tag))
drawrectangle(awesomeconf->display, phys_screen, drawrectangle(ctx, x, y,
x, y,
(awesomeconf->screens[screen].font->height + 2) / 4, (awesomeconf->screens[screen].font->height + 2) / 4,
(awesomeconf->screens[screen].font->height + 2) / 4, (awesomeconf->screens[screen].font->height + 2) / 4,
drawable,
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
sel && is_client_tagged(awesomeconf->screens[screen].tclink, sel, tag), sel && is_client_tagged(awesomeconf->screens[screen].tclink, sel, tag),
awesomeconf->screens[screen].colors_selected[ColFG]); awesomeconf->screens[screen].colors_selected[ColFG]);
} }
else else
{ {
drawtext(awesomeconf->display, phys_screen, drawtext(ctx, x, y, w,
x, y, w, sbar.height,
awesomeconf->screens[screen].statusbar.height,
drawable,
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
awesomeconf->screens[screen].font, awesomeconf->screens[screen].font,
tag->name, awesomeconf->screens[screen].colors_normal); tag->name,
awesomeconf->screens[screen].colors_normal);
if(isoccupied(awesomeconf->screens[screen].tclink, screen, awesomeconf->clients, tag)) if(isoccupied(awesomeconf->screens[screen].tclink, screen, awesomeconf->clients, tag))
drawrectangle(awesomeconf->display, phys_screen, drawrectangle(ctx, x, y,
x, y,
(awesomeconf->screens[screen].font->height + 2) / 4, (awesomeconf->screens[screen].font->height + 2) / 4,
(awesomeconf->screens[screen].font->height + 2) / 4, (awesomeconf->screens[screen].font->height + 2) / 4,
drawable,
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
sel && is_client_tagged(awesomeconf->screens[screen].tclink, sel, tag), sel && is_client_tagged(awesomeconf->screens[screen].tclink, sel, tag),
awesomeconf->screens[screen].colors_normal[ColFG]); awesomeconf->screens[screen].colors_normal[ColFG]);
} }
x += w; x += w;
} }
drawtext(awesomeconf->display, phys_screen,
x, y, awesomeconf->screens[screen].statusbar.txtlayoutwidth, /* This is only here until we refactor this function. */
awesomeconf->screens[screen].statusbar.height, for(layout = awesomeconf->screens[screen].layouts; layout; layout = layout->next)
drawable, sbar.txtlayoutwidth = MAX(sbar.txtlayoutwidth,
awesomeconf->screens[screen].statusbar.width, textwidth(ctx, awesomeconf->screens[screen].font, layout->symbol));
awesomeconf->screens[screen].statusbar.height, drawtext(ctx, x, y,
sbar.txtlayoutwidth,
sbar.height,
awesomeconf->screens[screen].font, awesomeconf->screens[screen].font,
get_current_layout(awesomeconf->screens[screen])->symbol, get_current_layout(awesomeconf->screens[screen])->symbol,
awesomeconf->screens[screen].colors_normal); awesomeconf->screens[screen].colors_normal);
z = x + awesomeconf->screens[screen].statusbar.txtlayoutwidth; z = x + sbar.txtlayoutwidth;
w = textwidth(awesomeconf->display, awesomeconf->screens[screen].font, awesomeconf->screens[screen].statustext); w = textwidth(ctx, awesomeconf->screens[screen].font, awesomeconf->screens[screen].statustext);
x = awesomeconf->screens[screen].statusbar.width - w; x = sbar.width - w;
if(x < z) if(x < z)
{ {
x = z; x = z;
w = awesomeconf->screens[screen].statusbar.width - z; w = sbar.width - z;
} }
drawtext(awesomeconf->display, phys_screen, drawtext(ctx, x, y, w,
x, y, w, sbar.height,
awesomeconf->screens[screen].statusbar.height,
drawable,
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
awesomeconf->screens[screen].font, awesomeconf->screens[screen].font,
awesomeconf->screens[screen].statustext, awesomeconf->screens[screen].colors_normal); awesomeconf->screens[screen].statustext,
if((w = x - z) > awesomeconf->screens[screen].statusbar.height) awesomeconf->screens[screen].colors_normal);
if((w = x - z) > sbar.height)
{ {
x = z; x = z;
if(sel && sel->screen == screen) if(sel && sel->screen == screen)
{ {
drawtext(awesomeconf->display, phys_screen, drawtext(ctx, x, y, w,
x, y, w, sbar.height,
awesomeconf->screens[screen].statusbar.height,
drawable,
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
awesomeconf->screens[screen].font, awesomeconf->screens[screen].font,
sel->name, awesomeconf->screens[screen].colors_selected); sel->name,
awesomeconf->screens[screen].colors_selected);
if(sel->isfloating) if(sel->isfloating)
drawcircle(awesomeconf->display, phys_screen, drawcircle(ctx, x, y,
x, y,
(awesomeconf->screens[screen].font->height + 2) / 4, (awesomeconf->screens[screen].font->height + 2) / 4,
drawable,
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
sel->ismax, sel->ismax,
awesomeconf->screens[screen].colors_selected[ColFG]); awesomeconf->screens[screen].colors_selected[ColFG]);
} }
else else
drawtext(awesomeconf->display, phys_screen, drawtext(ctx, x, y, w,
x, y, w, sbar.height,
awesomeconf->screens[screen].statusbar.height,
drawable,
awesomeconf->screens[screen].statusbar.width,
awesomeconf->screens[screen].statusbar.height,
awesomeconf->screens[screen].font, awesomeconf->screens[screen].font,
NULL, awesomeconf->screens[screen].colors_normal); NULL, awesomeconf->screens[screen].colors_normal);
} }
if(awesomeconf->screens[screen].statusbar.position == BarRight if(sbar.position == BarRight || sbar.position == BarLeft)
|| awesomeconf->screens[screen].statusbar.position == BarLeft)
{ {
Drawable d; Drawable d;
if(awesomeconf->screens[screen].statusbar.position == BarRight) if(sbar.position == BarRight)
d = draw_rotate(awesomeconf->display, phys_screen, drawable, d = draw_rotate(ctx, phys_screen, M_PI_2, sbar.height, 0);
awesomeconf->screens[screen].statusbar.width, awesomeconf->screens[screen].statusbar.height,
M_PI_2, awesomeconf->screens[screen].statusbar.height, 0);
else else
d = draw_rotate(awesomeconf->display, phys_screen, drawable, d = draw_rotate(ctx, phys_screen, - M_PI_2, 0, sbar.width);
awesomeconf->screens[screen].statusbar.width, awesomeconf->screens[screen].statusbar.height,
- M_PI_2, 0, awesomeconf->screens[screen].statusbar.width);
XCopyArea(awesomeconf->display, d, XCopyArea(awesomeconf->display, d,
awesomeconf->screens[screen].statusbar.window, sbar.window,
DefaultGC(awesomeconf->display, phys_screen), 0, 0, DefaultGC(awesomeconf->display, phys_screen), 0, 0,
awesomeconf->screens[screen].statusbar.height, awesomeconf->screens[screen].statusbar.width, 0, 0); sbar.height,
sbar.width, 0, 0);
XFreePixmap(awesomeconf->display, d); XFreePixmap(awesomeconf->display, d);
} }
else else
XCopyArea(awesomeconf->display, drawable, XCopyArea(awesomeconf->display, ctx->drawable,
awesomeconf->screens[screen].statusbar.window, sbar.window,
DefaultGC(awesomeconf->display, phys_screen), 0, 0, DefaultGC(awesomeconf->display, phys_screen), 0, 0,
awesomeconf->screens[screen].statusbar.width, awesomeconf->screens[screen].statusbar.height, 0, 0); sbar.width, sbar.height, 0, 0);
XFreePixmap(awesomeconf->display, drawable); draw_free_context(ctx);
XSync(awesomeconf->display, False); XSync(awesomeconf->display, False);
} }
void void
statusbar_init(Display *disp, int screen, Statusbar *statusbar, statusbar_init(Display *disp, int screen, Statusbar *statusbar, Cursor cursor, XftFont *font, Padding *padding)
Cursor cursor, XftFont *font, Layout *layouts, Padding *padding)
{ {
Layout *l;
XSetWindowAttributes wa; XSetWindowAttributes wa;
int phys_screen = get_phys_screen(disp, screen); int phys_screen = get_phys_screen(disp, screen);
ScreenInfo *si = get_screen_info(disp, screen, NULL, padding); ScreenInfo *si = get_screen_info(disp, screen, NULL, padding);
@ -241,10 +206,6 @@ statusbar_init(Display *disp, int screen, Statusbar *statusbar,
XDefineCursor(disp, statusbar->window, cursor); XDefineCursor(disp, statusbar->window, cursor);
statusbar_update_position(disp, *statusbar, padding); statusbar_update_position(disp, *statusbar, padding);
XMapRaised(disp, statusbar->window); XMapRaised(disp, statusbar->window);
for(l = layouts ; l; l = l->next)
statusbar->txtlayoutwidth = MAX(statusbar->txtlayoutwidth,
(textwidth(disp, font, l->symbol)));
} }
void void

View File

@ -24,7 +24,7 @@
#include "common.h" #include "common.h"
void statusbar_init(Display *, int, Statusbar *, Cursor, XftFont *, Layout *, Padding *); void statusbar_init(Display *, int, Statusbar *, Cursor, XftFont *, Padding *);
void statusbar_draw(awesome_config *, int); void statusbar_draw(awesome_config *, int);
int statusbar_get_position_from_str(const char *); int statusbar_get_position_from_str(const char *);
void statusbar_update_position(Display *, Statusbar, Padding *); void statusbar_update_position(Display *, Statusbar, Padding *);