diff --git a/awesomerc.5.txt b/awesomerc.5.txt index 1af42c91..f9fed856 100644 --- a/awesomerc.5.txt +++ b/awesomerc.5.txt @@ -44,6 +44,8 @@ general ^^^^^^^ General is a section containing some general options for this screen. +titlebar:: + This option defines the position of window titlebar. border:: This defines the window borders size in pixel. font:: diff --git a/client.c b/client.c index 165f48a3..3149429d 100644 --- a/client.c +++ b/client.c @@ -132,6 +132,39 @@ client_get_byname(Client *list, char *name) return NULL; } +static void +client_updatetitlebar(Client *c) +{ + DrawCtx *ctx; + int phys_screen; + style_t style; + area_t geometry; + + if(!c->titlebar) + return; + + phys_screen = get_phys_screen(c->screen); + + ctx = draw_context_new(globalconf.display, phys_screen, + c->titlebar->geometry.width, + c->titlebar->geometry.height, + c->titlebar->drawable); + + style = globalconf.focus->client == c ? + globalconf.screens[c->screen].styles.focus : + globalconf.screens[c->screen].styles.normal; + + geometry = c->titlebar->geometry; + geometry.x = geometry.y = 0; + + draw_text(ctx, geometry, AlignCenter, 0, + c->name, style); + + simplewindow_refresh_drawable(c->titlebar, phys_screen); + + draw_context_delete(ctx); +} + /** Update client name attribute with its title * \param c the client */ @@ -143,6 +176,8 @@ client_updatetitle(Client *c) xgettextprop(globalconf.display, c->win, XInternAtom(globalconf.display, "WM_NAME", False), c->name, sizeof(c->name)); + client_updatetitlebar(c); + widget_invalidate_cache(c->screen, WIDGET_CACHE_CLIENTS); } @@ -151,10 +186,11 @@ client_unfocus(Client *c) { if(globalconf.screens[c->screen].opacity_unfocused != -1) window_settrans(c->win, globalconf.screens[c->screen].opacity_unfocused); + focus_add_client(NULL); XSetWindowBorder(globalconf.display, c->win, globalconf.screens[c->screen].styles.normal.border.pixel); widget_invalidate_cache(c->screen, WIDGET_CACHE_CLIENTS); - focus_add_client(NULL); + client_updatetitlebar(c); } /** Ban client and unmap it @@ -167,6 +203,8 @@ client_ban(Client *c) client_unfocus(c); XUnmapWindow(globalconf.display, c->win); window_setstate(c->win, IconicState); + if(c->titlebar) + XUnmapWindow(globalconf.display, c->titlebar->window); } /** Give focus to client, or to first client if c is NULL @@ -201,13 +239,18 @@ client_focus(Client *c, int screen, Bool raise) window_settrans(c->win, -1); XSetWindowBorder(globalconf.display, c->win, globalconf.screens[screen].styles.focus.border.pixel); + client_updatetitlebar(c); XSetInputFocus(globalconf.display, c->win, RevertToPointerRoot, CurrentTime); if(raise) { XWindowChanges wc; Layout *curlay = layout_get_current(screen); if(c->isfloating || curlay->arrange == layout_floating) + { XRaiseWindow(globalconf.display, c->win); + if(c->titlebar) + XRaiseWindow(globalconf.display, c->titlebar->window); + } else { Client *client; @@ -216,14 +259,32 @@ client_focus(Client *c, int screen, Bool raise) for(client = globalconf.clients; client; client = client->next) if(client != c && client_isvisible(client, c->screen) && client->isfloating) { + if(client->titlebar) + { + XConfigureWindow(globalconf.display, client->titlebar->window, + CWSibling | CWStackMode, &wc); + wc.sibling = client->titlebar->window; + } XConfigureWindow(globalconf.display, client->win, CWSibling | CWStackMode, &wc); wc.sibling = client->win; } + if(c->titlebar) + { + XConfigureWindow(globalconf.display, c->titlebar->window, + CWSibling | CWStackMode, &wc); + wc.sibling = c->titlebar->window; + } XConfigureWindow(globalconf.display, c->win, CWSibling | CWStackMode, &wc); wc.sibling = c->win; for(client = globalconf.clients; client; client = client->next) if(client != c && IS_TILED(client, c->screen)) { + if(client->titlebar) + { + XConfigureWindow(globalconf.display, client->titlebar->window, + CWSibling | CWStackMode, &wc); + wc.sibling = client->titlebar->window; + } XConfigureWindow(globalconf.display, client->win, CWSibling | CWStackMode, &wc); wc.sibling = client->win; } @@ -257,7 +318,7 @@ client_manage(Window w, XWindowAttributes *wa, int screen) Tag *tag; Rule *rule; area_t screen_geom; - int phys_screen = get_phys_screen(screen); + int phys_screen = get_phys_screen(screen), titlebar_height; long flags; c = p_new(Client, 1); @@ -286,6 +347,23 @@ client_manage(Window w, XWindowAttributes *wa, int screen) /* propagates border_width, if size doesn't change */ window_configure(c->win, c->geometry, c->border); + switch(globalconf.screens[c->screen].titlebar) + { + case Top: + titlebar_height = 1.5 * MAX(globalconf.screens[c->screen].styles.normal.font->height, + MAX(globalconf.screens[c->screen].styles.focus.font->height, + globalconf.screens[c->screen].styles.urgent.font->height)), + c->titlebar = simplewindow_new(globalconf.display, + phys_screen, + c->geometry.x, + c->geometry.y - titlebar_height, + c->geometry.width, + titlebar_height, + 0); + break; + default: + break; + } /* update window title */ client_updatetitle(c); @@ -387,6 +465,53 @@ client_manage(Window w, XWindowAttributes *wa, int screen) ewmh_update_net_client_list(phys_screen); } +static area_t +client_geometry_sizehint(Client *c, area_t geometry) +{ + double dx, dy, max, min, ratio; + + if(c->minay > 0 && c->maxay > 0 && (geometry.height - c->baseh) > 0 + && (geometry.width - c->basew) > 0) + { + dx = (double) (geometry.width - c->basew); + dy = (double) (geometry.height - c->baseh); + min = (double) (c->minax) / (double) (c->minay); + max = (double) (c->maxax) / (double) (c->maxay); + ratio = dx / dy; + if(max > 0 && min > 0 && ratio > 0) + { + if(ratio < min) + { + dy = (dx * min + dy) / (min * min + 1); + dx = dy * min; + geometry.width = (int) dx + c->basew; + geometry.height = (int) dy + c->baseh; + } + else if(ratio > max) + { + dy = (dx * min + dy) / (max * max + 1); + dx = dy * min; + geometry.width = (int) dx + c->basew; + geometry.height = (int) dy + c->baseh; + } + } + } + if(c->minw && geometry.width < c->minw) + geometry.width = c->minw; + if(c->minh && geometry.height < c->minh) + geometry.height = c->minh; + if(c->maxw && geometry.width > c->maxw) + geometry.width = c->maxw; + if(c->maxh && geometry.height > c->maxh) + geometry.height = c->maxh; + if(c->incw) + geometry.width -= (geometry.width - c->basew) % c->incw; + if(c->inch) + geometry.height -= (geometry.height - c->baseh) % c->inch; + + return geometry; +} + /** Resize client window * \param c client to resize * \param geometry new window geometry @@ -397,53 +522,21 @@ Bool client_resize(Client *c, area_t geometry, Bool sizehints) { int new_screen; - double dx, dy, max, min, ratio; area_t area; XWindowChanges wc; - if(sizehints) + if(c->titlebar) { - if(c->minay > 0 && c->maxay > 0 && (geometry.height - c->baseh) > 0 - && (geometry.width - c->basew) > 0) - { - dx = (double) (geometry.width - c->basew); - dy = (double) (geometry.height - c->baseh); - min = (double) (c->minax) / (double) (c->minay); - max = (double) (c->maxax) / (double) (c->maxay); - ratio = dx / dy; - if(max > 0 && min > 0 && ratio > 0) - { - if(ratio < min) - { - dy = (dx * min + dy) / (min * min + 1); - dx = dy * min; - geometry.width = (int) dx + c->basew; - geometry.height = (int) dy + c->baseh; - } - else if(ratio > max) - { - dy = (dx * min + dy) / (max * max + 1); - dx = dy * min; - geometry.width = (int) dx + c->basew; - geometry.height = (int) dy + c->baseh; - } - } - } - if(c->minw && geometry.width < c->minw) - geometry.width = c->minw; - if(c->minh && geometry.height < c->minh) - geometry.height = c->minh; - if(c->maxw && geometry.width > c->maxw) - geometry.width = c->maxw; - if(c->maxh && geometry.height > c->maxh) - geometry.height = c->maxh; - if(c->incw) - geometry.width -= (geometry.width - c->basew) % c->incw; - if(c->inch) - geometry.height -= (geometry.height - c->baseh) % c->inch; + geometry.y += c->titlebar->geometry.height; + geometry.height -= c->titlebar->geometry.height; } + + if(sizehints) + geometry = client_geometry_sizehint(c, geometry); + if(geometry.width <= 0 || geometry.height <= 0) return False; + /* offscreen appearance fixes */ area = get_display_area(get_phys_screen(c->screen), NULL, @@ -463,10 +556,22 @@ client_resize(Client *c, area_t geometry, Bool sizehints) new_screen = screen_get_bycoord(globalconf.screens_info, c->screen, geometry.x, geometry.y); c->geometry.x = wc.x = geometry.x; - c->geometry.y = wc.y = geometry.y; c->geometry.width = wc.width = geometry.width; + c->geometry.y = wc.y = geometry.y; c->geometry.height = wc.height = geometry.height; - wc.border_width = c->border; + + if(c->titlebar) + { + simplewindow_move_resize(c->titlebar, + geometry.x, + geometry.y - c->titlebar->geometry.height, + geometry.width, + c->titlebar->geometry.height); + client_updatetitlebar(c); + + c->geometry.y -= c->titlebar->geometry.height; + c->geometry.height += c->titlebar->geometry.height; + } /* save the floating geometry if the window is floating but not * maximized */ @@ -474,8 +579,10 @@ client_resize(Client *c, area_t geometry, Bool sizehints) layout_get_current(new_screen)->arrange == layout_floating) && !c->ismax) c->f_geometry = geometry; + printf("moving client %s to %d\n", c->name, c->geometry.y); + XConfigureWindow(globalconf.display, c->win, - CWX | CWY | CWWidth | CWHeight | CWBorderWidth, &wc); + CWX | CWY | CWWidth | CWHeight, &wc); window_configure(c->win, geometry, c->border); if(c->screen != new_screen) @@ -547,6 +654,8 @@ client_unban(Client *c) { XMapWindow(globalconf.display, c->win); window_setstate(c->win, NormalState); + if(c->titlebar) + XMapWindow(globalconf.display, c->titlebar->window); } void @@ -579,6 +688,9 @@ client_unmanage(Client *c) XSync(globalconf.display, False); XUngrabServer(globalconf.display); + if(c->titlebar) + simplewindow_delete(c->titlebar); + p_delete(&c); } diff --git a/common/configopts.c b/common/configopts.c index a0107674..5d68afe1 100644 --- a/common/configopts.c +++ b/common/configopts.c @@ -70,6 +70,7 @@ cfg_awesome_include(cfg_t *cfg, cfg_opt_t *opt, cfg_opt_t general_opts[] = { CFG_INT((char *) "border", 1, CFGF_NONE), + CFG_STR((char *) "titlebar", (char *) "off", CFGF_NONE), CFG_INT((char *) "snap", 8, CFGF_NONE), CFG_BOOL((char *) "resize_hints", cfg_true, CFGF_NONE), CFG_BOOL((char *) "sloppy_focus", cfg_true, CFGF_NONE), diff --git a/common/util.c b/common/util.c index 7eff4fb9..d09e1f7a 100644 --- a/common/util.c +++ b/common/util.c @@ -96,6 +96,20 @@ name_func_lookup(const char *funcname, const name_func_link_t *list) return NULL; } +Position +position_get_from_str(const char *pos) +{ + if(!a_strncmp(pos, "top", 3)) + return Top; + else if(!a_strncmp(pos, "bottom", 6)) + return Bottom; + else if(!a_strncmp(pos, "right", 5)) + return Right; + else if(!a_strncmp(pos, "left", 4)) + return Left; + return Off; +} + /** \brief safe limited strcpy. * * Copies at most min(n-1, \c l) characters from \c src into \c dst, diff --git a/common/util.h b/common/util.h index cba63238..93220853 100644 --- a/common/util.h +++ b/common/util.h @@ -26,6 +26,16 @@ #include #include +/** A list of possible position, not sex related */ +typedef enum +{ + Off = 0, + Top, + Bottom, + Right, + Left +} Position; + /** Link a name to a function */ typedef struct { @@ -228,6 +238,7 @@ void _eprint(int, const char *, const char *, ...) void _warn(int, const char *, const char *, ...) __attribute__ ((format(printf, 3, 4))); +Position position_get_from_str(const char *); double compute_new_value_from_arg(const char *, double); void *name_func_lookup(const char *, const name_func_link_t *); diff --git a/config.c b/config.c index 369f4219..46661b34 100644 --- a/config.c +++ b/config.c @@ -319,6 +319,7 @@ config_parse_screen(cfg_t *cfg, int screen) virtscreen->floating_placement = name_func_lookup(cfg_getstr(cfg_general, "floating_placement"), FloatingPlacementList); + virtscreen->titlebar = position_get_from_str(cfg_getstr(cfg_general, "titlebar")); virtscreen->mwfact_lower_limit = cfg_getfloat(cfg_general, "mwfact_lower_limit"); virtscreen->mwfact_upper_limit = cfg_getfloat(cfg_general, "mwfact_upper_limit"); @@ -378,7 +379,7 @@ config_parse_screen(cfg_t *cfg, int screen) statusbar = p_new(Statusbar, 1); cfgsectmp = cfg_getnsec(cfg_screen, "statusbar", i); statusbar->position = statusbar->dposition = - statusbar_get_position_from_str(cfg_getstr(cfgsectmp, "position")); + position_get_from_str(cfg_getstr(cfgsectmp, "position")); statusbar->height = cfg_getint(cfgsectmp, "height"); statusbar->width = cfg_getint(cfgsectmp, "width"); statusbar->name = a_strdup(cfg_title(cfgsectmp)); diff --git a/event.c b/event.c index 25424821..46d67ce8 100644 --- a/event.c +++ b/event.c @@ -289,8 +289,10 @@ event_handle_expose(XEvent *e) XExposeEvent *ev = &e->xexpose; int screen; Statusbar *statusbar; + Client *c; if(!ev->count) + { for(screen = 0; screen < globalconf.screens_info->nscreen; screen++) for(statusbar = globalconf.screens[screen].statusbar; statusbar; statusbar = statusbar->next) if(statusbar->sw->window == ev->window) @@ -298,6 +300,14 @@ event_handle_expose(XEvent *e) statusbar_display(statusbar); return; } + + for(c = globalconf.clients; c; c = c->next) + if(c->titlebar && c->titlebar->window == ev->window) + { + simplewindow_refresh_drawable(c->titlebar, get_phys_screen(c->screen)); + return; + } + } } /** Handle XKey events diff --git a/statusbar.c b/statusbar.c index b45dce31..fabb411e 100644 --- a/statusbar.c +++ b/statusbar.c @@ -244,20 +244,6 @@ statusbar_refresh() } } -Position -statusbar_get_position_from_str(const char *pos) -{ - if(!a_strncmp(pos, "off", 3)) - return Off; - else if(!a_strncmp(pos, "bottom", 6)) - return Bottom; - else if(!a_strncmp(pos, "right", 5)) - return Right; - else if(!a_strncmp(pos, "left", 4)) - return Left; - return Top; -} - static Statusbar * get_statusbar_byname(int screen, const char *name) { diff --git a/statusbar.h b/statusbar.h index b0841c9d..b0759d6a 100644 --- a/statusbar.h +++ b/statusbar.h @@ -28,7 +28,6 @@ void statusbar_refresh(void); void statusbar_preinit(Statusbar *); void statusbar_init(Statusbar *); void statusbar_display(Statusbar *); -Position statusbar_get_position_from_str(const char *); Uicb uicb_statusbar_toggle; diff --git a/structs.h b/structs.h index 385c4940..1c3a6d22 100644 --- a/structs.h +++ b/structs.h @@ -28,16 +28,6 @@ #include "common/swindow.h" #include "common/xscreen.h" -/** Bar possible position */ -typedef enum -{ - Top, - Bottom, - Left, - Right, - Off -} Position; - /** Rules for floating rule */ typedef enum { @@ -200,6 +190,8 @@ struct Client int screen; /** True if the client is a new one */ Bool newcomer; + /** Titlebar */ + SimpleWindow *titlebar; }; typedef struct client_node_t client_node_t; @@ -260,6 +252,8 @@ typedef struct typedef area_t (FloatingPlacement)(area_t, int, int); typedef struct { + /** Titlebar position */ + Position titlebar; /** Number of pixels to snap windows */ int snap; /** Border size */