Merge commit '69513d6'

Conflicts:
	src/compton.c
	src/compton.h
This commit is contained in:
Christopher Jeffrey 2012-09-28 12:22:25 -05:00
commit 7c1838cfd8
3 changed files with 269 additions and 217 deletions

View File

@ -24,9 +24,11 @@ fading = true;
# fade-delta = 30 # fade-delta = 30
fade-in-step = 0.03 fade-in-step = 0.03
fade-out-step = 0.03 fade-out-step = 0.03
# no-fading-openclose = true;
# Other # Other
mark-wmwin-focused = true; mark-wmwin-focused = true;
mark-ovredir-focused = true;
# Window type settings # Window type settings
wintypes: wintypes:

View File

@ -96,11 +96,6 @@ Atom class_atom;
Atom win_type_atom; Atom win_type_atom;
Atom win_type[NUM_WINTYPES]; Atom win_type[NUM_WINTYPES];
// Window type settings
double win_type_opacity[NUM_WINTYPES];
Bool win_type_shadow[NUM_WINTYPES];
Bool win_type_fade[NUM_WINTYPES];
/** /**
* Macros * Macros
*/ */
@ -112,7 +107,7 @@ Bool win_type_fade[NUM_WINTYPES];
* Options * Options
*/ */
static options_t options = { static options_t opts = {
.display = NULL, .display = NULL,
.shadow_radius = 12, .shadow_radius = 12,
.shadow_offset_x = -15, .shadow_offset_x = -15,
@ -121,27 +116,26 @@ static options_t options = {
.fade_in_step = 0.028 * OPAQUE, .fade_in_step = 0.028 * OPAQUE,
.fade_out_step = 0.03 * OPAQUE, .fade_out_step = 0.03 * OPAQUE,
.fade_delta = 10, .fade_delta = 10,
.fade_trans = False, .no_fading_openclose = False,
.clear_shadow = False, .clear_shadow = False,
.inactive_opacity = 0, .inactive_opacity = 0,
.inactive_opacity_override = False, .inactive_opacity_override = False,
.frame_opacity = 0.0, .frame_opacity = 0.0,
.inactive_dim = 0.0, .inactive_dim = 0.0,
.mark_wmwin_focused = False, .mark_wmwin_focused = False,
.track_focus = False, .mark_ovredir_focused = False,
.track_wdata = False,
.shadow_blacklist = NULL, .shadow_blacklist = NULL,
.fade_blacklist = NULL, .fade_blacklist = NULL,
.fork_after_register = False, .fork_after_register = False,
.shadow_red = 0.0, .shadow_red = 0.0,
.shadow_green = 0.0, .shadow_green = 0.0,
.shadow_blue = 0.0, .shadow_blue = 0.0,
.wintype_opacity = { 0.0 },
.wintype_shadow = { False },
.wintype_fade = { False },
.synchronize = False, .synchronize = False,
.shadow_enable = 0, .track_focus = False,
.fading_enable = 0, .track_wdata = False,
.no_dock_shadow = False,
.no_dnd_shadow = False,
.menu_opacity = 1.0
}; };
/** /**
@ -172,7 +166,7 @@ get_time_in_milliseconds() {
*/ */
static int static int
fade_timeout(void) { fade_timeout(void) {
int diff = options.fade_delta - get_time_in_milliseconds() + fade_time; int diff = opts.fade_delta - get_time_in_milliseconds() + fade_time;
if (diff < 0) if (diff < 0)
diff = 0; diff = 0;
@ -201,11 +195,11 @@ run_fade(Display *dpy, win *w, unsigned steps) {
// calculations // calculations
if (w->opacity < w->opacity_tgt) if (w->opacity < w->opacity_tgt)
w->opacity = normalize_d_range( w->opacity = normalize_d_range(
(double) w->opacity + (double) options.fade_in_step * steps, (double) w->opacity + (double) opts.fade_in_step * steps,
0.0, w->opacity_tgt); 0.0, w->opacity_tgt);
else else
w->opacity = normalize_d_range( w->opacity = normalize_d_range(
(double) w->opacity - (double) options.fade_out_step * steps, (double) w->opacity - (double) opts.fade_out_step * steps,
w->opacity_tgt, OPAQUE); w->opacity_tgt, OPAQUE);
} }
@ -421,12 +415,12 @@ make_shadow(Display *dpy, double opacity,
* center (fill the complete data array) * center (fill the complete data array)
*/ */
// If options.clear_shadow is enabled and the border & corner shadow (which // If clear_shadow is enabled and the border & corner shadow (which
// later will be filled) could entirely cover the area of the shadow // later will be filled) could entirely cover the area of the shadow
// that will be displayed, do not bother filling other pixels. If it // that will be displayed, do not bother filling other pixels. If it
// can't, we must fill the other pixels here. // can't, we must fill the other pixels here.
if (!(options.clear_shadow && options.shadow_offset_x <= 0 && options.shadow_offset_x >= -cgsize if (!(opts.clear_shadow && opts.shadow_offset_x <= 0 && opts.shadow_offset_x >= -cgsize
&& options.shadow_offset_y <= 0 && options.shadow_offset_y >= -cgsize)) { && opts.shadow_offset_y <= 0 && opts.shadow_offset_y >= -cgsize)) {
if (cgsize > 0) { if (cgsize > 0) {
d = shadow_top[opacity_int * (cgsize + 1) + cgsize]; d = shadow_top[opacity_int * (cgsize + 1) + cgsize];
} else { } else {
@ -497,14 +491,14 @@ make_shadow(Display *dpy, double opacity,
} }
} }
if (options.clear_shadow) { if (opts.clear_shadow) {
// Clear the region in the shadow that the window would cover based // Clear the region in the shadow that the window would cover based
// on shadow_offset_{x,y} user provides // on shadow_offset_{x,y} user provides
int xstart = normalize_i_range(- (int) options.shadow_offset_x, 0, swidth); int xstart = normalize_i_range(- (int) opts.shadow_offset_x, 0, swidth);
int xrange = normalize_i_range(width - (int) options.shadow_offset_x, int xrange = normalize_i_range(width - (int) opts.shadow_offset_x,
0, swidth) - xstart; 0, swidth) - xstart;
int ystart = normalize_i_range(- (int) options.shadow_offset_y, 0, sheight); int ystart = normalize_i_range(- (int) opts.shadow_offset_y, 0, sheight);
int yend = normalize_i_range(height - (int) options.shadow_offset_y, int yend = normalize_i_range(height - (int) opts.shadow_offset_y,
0, sheight); 0, sheight);
int y; int y;
@ -872,11 +866,11 @@ determine_evmask(Display *dpy, Window wid, win_evmode_t mode) {
if (WIN_EVMODE_FRAME == mode || find_win(dpy, wid)) { if (WIN_EVMODE_FRAME == mode || find_win(dpy, wid)) {
evmask |= PropertyChangeMask; evmask |= PropertyChangeMask;
if (options.track_focus) evmask |= FocusChangeMask; if (opts.track_focus) evmask |= FocusChangeMask;
} }
if (WIN_EVMODE_CLIENT == mode || find_toplevel(dpy, wid)) { if (WIN_EVMODE_CLIENT == mode || find_toplevel(dpy, wid)) {
if (options.frame_opacity || options.track_wdata) if (opts.frame_opacity || opts.track_wdata)
evmask |= PropertyChangeMask; evmask |= PropertyChangeMask;
} }
@ -1178,7 +1172,7 @@ paint_preprocess(Display *dpy, win *list) {
// Sounds like the timeout in poll() frequently does not work // Sounds like the timeout in poll() frequently does not work
// accurately, asking it to wait to 20ms, and often it would wait for // accurately, asking it to wait to 20ms, and often it would wait for
// 19ms, so the step value has to be rounded. // 19ms, so the step value has to be rounded.
unsigned steps = roundl((double) (get_time_in_milliseconds() - fade_time) / options.fade_delta); unsigned steps = roundl((double) (get_time_in_milliseconds() - fade_time) / opts.fade_delta);
// Reset fade_time // Reset fade_time
fade_time = get_time_in_milliseconds(); fade_time = get_time_in_milliseconds();
@ -1256,9 +1250,9 @@ paint_preprocess(Display *dpy, win *list) {
w->opacity_cur = w->opacity; w->opacity_cur = w->opacity;
} }
// Calculate options.frame_opacity // Calculate frame_opacity
if (options.frame_opacity && 1.0 != options.frame_opacity && w->top_width) if (opts.frame_opacity && 1.0 != opts.frame_opacity && w->top_width)
w->frame_opacity = get_opacity_percent(dpy, w) * options.frame_opacity; w->frame_opacity = get_opacity_percent(dpy, w) * opts.frame_opacity;
else else
w->frame_opacity = 0.0; w->frame_opacity = 0.0;
@ -1274,9 +1268,9 @@ paint_preprocess(Display *dpy, win *list) {
// Calculate shadow opacity // Calculate shadow opacity
if (w->frame_opacity) if (w->frame_opacity)
w->shadow_opacity = options.shadow_opacity * w->frame_opacity; w->shadow_opacity = opts.shadow_opacity * w->frame_opacity;
else else
w->shadow_opacity = options.shadow_opacity * get_opacity_percent(dpy, w); w->shadow_opacity = opts.shadow_opacity * get_opacity_percent(dpy, w);
// Rebuild shadow_pict if necessary // Rebuild shadow_pict if necessary
if (w->flags & WFLAG_SIZE_CHANGE) if (w->flags & WFLAG_SIZE_CHANGE)
@ -1562,7 +1556,7 @@ map_win(Display *dpy, Window id,
mark_client_win(dpy, w, cw); mark_client_win(dpy, w, cw);
} }
} }
else if (options.frame_opacity) { else if (opts.frame_opacity) {
// Refetch frame extents just in case it changes when the window is // Refetch frame extents just in case it changes when the window is
// unmapped // unmapped
get_frame_extents(dpy, w, w->client_win); get_frame_extents(dpy, w, w->client_win);
@ -1577,7 +1571,7 @@ map_win(Display *dpy, Window id,
#endif #endif
// Get window name and class if we are tracking them // Get window name and class if we are tracking them
if (options.track_wdata) { if (opts.track_wdata) {
win_get_name(dpy, w); win_get_name(dpy, w);
win_get_class(dpy, w); win_get_class(dpy, w);
} }
@ -1588,24 +1582,35 @@ map_win(Display *dpy, Window id,
* XSelectInput() is called too late. We have to recheck the focused * XSelectInput() is called too late. We have to recheck the focused
* window here. * window here.
*/ */
if (options.track_focus) { if (opts.track_focus) {
recheck_focus(dpy); recheck_focus(dpy);
// Consider a window without client window a WM window and mark it // Consider a window without client window a WM window and mark it
// focused if options.mark_wmwin_focused is on // focused if mark_wmwin_focused is on, or it's over-redirected and
if (options.mark_wmwin_focused && !w->client_win) // mark_ovredir_focused is on
if ((opts.mark_wmwin_focused && !w->client_win)
|| (opts.mark_ovredir_focused && w->a.override_redirect))
w->focused = True; w->focused = True;
} }
// Window type change could affect shadow and fade // Window type change could affect shadow and fade
determine_shadow(dpy, w); determine_shadow(dpy, w);
determine_fade(dpy, w);
// Determine mode here just in case the colormap changes // Determine mode here just in case the colormap changes
determine_mode(dpy, w); determine_mode(dpy, w);
// Fading in // Fading in
calc_opacity(dpy, w, True); calc_opacity(dpy, w, True);
set_fade_callback(dpy, w, NULL, True);
if (opts.no_fading_openclose) {
set_fade_callback(dpy, w, finish_map_win, True);
// Must be set after we execute the old fade callback, in case we
// receive two continuous MapNotify for the same window
w->fade = False;
}
else {
set_fade_callback(dpy, w, NULL, True);
determine_fade(dpy, w);
}
calc_dim(dpy, w); calc_dim(dpy, w);
@ -1624,6 +1629,12 @@ map_win(Display *dpy, Window id,
} }
} }
static void
finish_map_win(Display *dpy, win *w) {
if (opts.no_fading_openclose)
determine_fade(dpy, w);
}
static void static void
finish_unmap_win(Display *dpy, win *w) { finish_unmap_win(Display *dpy, win *w) {
w->damaged = 0; w->damaged = 0;
@ -1662,6 +1673,8 @@ unmap_win(Display *dpy, Window id, Bool fade) {
// Fading out // Fading out
w->opacity_tgt = 0; w->opacity_tgt = 0;
set_fade_callback(dpy, w, unmap_callback, False); set_fade_callback(dpy, w, unmap_callback, False);
if (opts.no_fading_openclose)
w->fade = False;
// don't care about properties anymore // don't care about properties anymore
// Will get BadWindow if the window is destroyed // Will get BadWindow if the window is destroyed
@ -1727,17 +1740,17 @@ determine_mode(Display *dpy, win *w) {
/** /**
* Calculate and set the opacity of a window. * Calculate and set the opacity of a window.
* *
* If window is inactive and options.inactive_opacity_override is set, the * If window is inactive and inactive_opacity_override is set, the
* priority is: (Simulates the old behavior) * priority is: (Simulates the old behavior)
* *
* options.inactive_opacity > _NET_WM_WINDOW_OPACITY (if not opaque) * inactive_opacity > _NET_WM_WINDOW_OPACITY (if not opaque)
* > window type default opacity * > window type default opacity
* *
* Otherwise: * Otherwise:
* *
* _NET_WM_WINDOW_OPACITY (if not opaque) * _NET_WM_WINDOW_OPACITY (if not opaque)
* > window type default opacity (if not opaque) * > window type default opacity (if not opaque)
* > options.inactive_opacity * > inactive_opacity
* *
* @param dpy X display to use * @param dpy X display to use
* @param w struct _win object representing the window * @param w struct _win object representing the window
@ -1760,15 +1773,15 @@ calc_opacity(Display *dpy, win *w, Bool refetch_prop) {
} }
if (OPAQUE == (opacity = w->opacity_prop)) { if (OPAQUE == (opacity = w->opacity_prop)) {
if (1.0 != win_type_opacity[w->window_type]) { if (1.0 != opts.wintype_opacity[w->window_type]) {
opacity = win_type_opacity[w->window_type] * OPAQUE; opacity = opts.wintype_opacity[w->window_type] * OPAQUE;
} }
} }
// Respect options.inactive_opacity in some cases // Respect inactive_opacity in some cases
if (options.inactive_opacity && is_normal_win(w) && False == w->focused if (opts.inactive_opacity && is_normal_win(w) && False == w->focused
&& (OPAQUE == opacity || options.inactive_opacity_override)) { && (OPAQUE == opacity || opts.inactive_opacity_override)) {
opacity = options.inactive_opacity; opacity = opts.inactive_opacity;
} }
w->opacity_tgt = opacity; w->opacity_tgt = opacity;
@ -1778,7 +1791,7 @@ static void
calc_dim(Display *dpy, win *w) { calc_dim(Display *dpy, win *w) {
Bool dim; Bool dim;
if (options.inactive_dim && is_normal_win(w) && !(w->focused)) { if (opts.inactive_dim && is_normal_win(w) && !(w->focused)) {
dim = True; dim = True;
} else { } else {
dim = False; dim = False;
@ -1795,7 +1808,7 @@ calc_dim(Display *dpy, win *w) {
*/ */
static void static void
determine_fade(Display *dpy, win *w) { determine_fade(Display *dpy, win *w) {
w->fade = win_type_fade[w->window_type]; w->fade = opts.wintype_fade[w->window_type];
} }
/** /**
@ -1806,8 +1819,8 @@ static void
determine_shadow(Display *dpy, win *w) { determine_shadow(Display *dpy, win *w) {
Bool shadow_old = w->shadow; Bool shadow_old = w->shadow;
w->shadow = (win_type_shadow[w->window_type] w->shadow = (opts.wintype_shadow[w->window_type]
&& !win_match(w, options.shadow_blacklist, &w->cache_sblst)); && !win_match(w, opts.shadow_blacklist, &w->cache_sblst));
// Window extents need update on shadow state change // Window extents need update on shadow state change
if (w->shadow != shadow_old) { if (w->shadow != shadow_old) {
@ -1844,8 +1857,8 @@ calc_win_size(Display *dpy, win *w) {
*/ */
static void static void
calc_shadow_geometry(Display *dpy, win *w) { calc_shadow_geometry(Display *dpy, win *w) {
w->shadow_dx = options.shadow_offset_x; w->shadow_dx = opts.shadow_offset_x;
w->shadow_dy = options.shadow_offset_y; w->shadow_dy = opts.shadow_offset_y;
w->shadow_width = w->widthb + gaussian_map->size; w->shadow_width = w->widthb + gaussian_map->size;
w->shadow_height = w->heightb + gaussian_map->size; w->shadow_height = w->heightb + gaussian_map->size;
} }
@ -1863,7 +1876,7 @@ mark_client_win(Display *dpy, win *w, Window client) {
// Get the frame width and monitor further frame width changes on client // Get the frame width and monitor further frame width changes on client
// window if necessary // window if necessary
if (options.frame_opacity) { if (opts.frame_opacity) {
get_frame_extents(dpy, w, client); get_frame_extents(dpy, w, client);
} }
XSelectInput(dpy, client, determine_evmask(dpy, client, WIN_EVMODE_CLIENT)); XSelectInput(dpy, client, determine_evmask(dpy, client, WIN_EVMODE_CLIENT));
@ -2200,9 +2213,9 @@ damage_win(Display *dpy, XDamageNotifyEvent *de) {
&& w->damage_bounds.y <= 0 && w->damage_bounds.y <= 0
&& w->a.width <= w->damage_bounds.x + w->damage_bounds.width && w->a.width <= w->damage_bounds.x + w->damage_bounds.width
&& w->a.height <= w->damage_bounds.y + w->damage_bounds.height) { && w->a.height <= w->damage_bounds.y + w->damage_bounds.height) {
if (win_type_fade[w->window_type]) { if (opts.wintype_fade[w->window_type]) {
set_fade(dpy, w, 0, get_opacity_percent(dpy, w), set_fade(dpy, w, 0, get_opacity_percent(dpy, w),
options.fade_in_step, 0, True, True); opts.fade_in_step, 0, True, True);
} }
w->usable = True; w->usable = True;
} }
@ -2687,7 +2700,7 @@ ev_property_notify(XPropertyEvent *ev) {
} }
// If frame extents property changes // If frame extents property changes
if (options.frame_opacity && ev->atom == extents_atom) { if (opts.frame_opacity && ev->atom == extents_atom) {
win *w = find_toplevel(dpy, ev->window); win *w = find_toplevel(dpy, ev->window);
if (w) { if (w) {
get_frame_extents(dpy, w, ev->window); get_frame_extents(dpy, w, ev->window);
@ -2697,7 +2710,7 @@ ev_property_notify(XPropertyEvent *ev) {
} }
// If name changes // If name changes
if (options.track_wdata if (opts.track_wdata
&& (name_atom == ev->atom || name_ewmh_atom == ev->atom)) { && (name_atom == ev->atom || name_ewmh_atom == ev->atom)) {
win *w = find_toplevel(dpy, ev->window); win *w = find_toplevel(dpy, ev->window);
if (w && 1 == win_get_name(dpy, w)) if (w && 1 == win_get_name(dpy, w))
@ -2705,7 +2718,7 @@ ev_property_notify(XPropertyEvent *ev) {
} }
// If class changes // If class changes
if (options.track_wdata && class_atom == ev->atom) { if (opts.track_wdata && class_atom == ev->atom) {
win *w = find_toplevel(dpy, ev->window); win *w = find_toplevel(dpy, ev->window);
if (w) { if (w) {
win_get_class(dpy, w); win_get_class(dpy, w);
@ -2858,9 +2871,10 @@ usage(void) {
"-z\n" "-z\n"
" Zero the part of the shadow's mask behind the window (experimental).\n" " Zero the part of the shadow's mask behind the window (experimental).\n"
"-f\n" "-f\n"
" Fade windows in/out when opening/closing.\n" " Fade windows in/out when opening/closing and when opacity\n"
" changes, unless --no-fading-openclose is used.\n"
"-F\n" "-F\n"
" Fade windows during opacity changes.\n" " Equals -f. Deprecated.\n"
"-i opacity\n" "-i opacity\n"
" Opacity of inactive windows. (0.1 - 1.0)\n" " Opacity of inactive windows. (0.1 - 1.0)\n"
"-e opacity\n" "-e opacity\n"
@ -2887,6 +2901,10 @@ usage(void) {
" Try to detect WM windows and mark them as active.\n" " Try to detect WM windows and mark them as active.\n"
"--shadow-exclude condition\n" "--shadow-exclude condition\n"
" Exclude conditions for shadows.\n" " Exclude conditions for shadows.\n"
"--mark-ovredir-focused\n"
" Mark over-redirect windows as active.\n"
"--no-fading-openclose\n"
" Do not fade on window open/close.\n"
"\n" "\n"
"Format of a condition:\n" "Format of a condition:\n"
"\n" "\n"
@ -3052,7 +3070,7 @@ open_config_file(char *cpath, char **ppath) {
* Parse a configuration file from default location. * Parse a configuration file from default location.
*/ */
static void static void
parse_config(char *cpath) { parse_config(char *cpath, struct options_tmp *pcfgtmp) {
char *path = NULL, *parent = NULL; char *path = NULL, *parent = NULL;
FILE *f; FILE *f;
config_t cfg; config_t cfg;
@ -3085,63 +3103,60 @@ parse_config(char *cpath) {
// Get options from the configuration file. We don't do range checking // Get options from the configuration file. We don't do range checking
// right now. It will be done later // right now. It will be done later
// -D (options.fade_delta) // -D (fade_delta)
if (config_lookup_int(&cfg, "fade-delta", &ival)) if (config_lookup_int(&cfg, "fade-delta", &ival))
options.fade_delta = ival; opts.fade_delta = ival;
// -I (options.fade_in_step) // -I (fade_in_step)
if (config_lookup_float(&cfg, "fade-in-step", &dval)) if (config_lookup_float(&cfg, "fade-in-step", &dval))
options.fade_in_step = normalize_d(dval) * OPAQUE; opts.fade_in_step = normalize_d(dval) * OPAQUE;
// -O (options.fade_out_step) // -O (fade_out_step)
if (config_lookup_float(&cfg, "fade-out-step", &dval)) if (config_lookup_float(&cfg, "fade-out-step", &dval))
options.fade_out_step = normalize_d(dval) * OPAQUE; opts.fade_out_step = normalize_d(dval) * OPAQUE;
// -r (options.shadow_radius) // -r (shadow_radius)
config_lookup_int(&cfg, "shadow-radius", &options.shadow_radius); config_lookup_int(&cfg, "shadow-radius", &opts.shadow_radius);
// -o (options.shadow_opacity) // -o (shadow_opacity)
config_lookup_float(&cfg, "shadow-opacity", &options.shadow_opacity); config_lookup_float(&cfg, "shadow-opacity", &opts.shadow_opacity);
// -l (options.shadow_offset_x) // -l (shadow_offset_x)
config_lookup_int(&cfg, "shadow-offset-x", &options.shadow_offset_x); config_lookup_int(&cfg, "shadow-offset-x", &opts.shadow_offset_x);
// -t (options.shadow_offset_y) // -t (shadow_offset_y)
config_lookup_int(&cfg, "shadow-offset-y", &options.shadow_offset_y); config_lookup_int(&cfg, "shadow-offset-y", &opts.shadow_offset_y);
// -i (options.inactive_opacity) // -i (inactive_opacity)
if (config_lookup_float(&cfg, "inactive-opacity", &dval)) if (config_lookup_float(&cfg, "inactive-opacity", &dval))
options.inactive_opacity = normalize_d(dval) * OPAQUE; opts.inactive_opacity = normalize_d(dval) * OPAQUE;
// -e (options.frame_opacity) // -e (frame_opacity)
config_lookup_float(&cfg, "frame-opacity", &options.frame_opacity); config_lookup_float(&cfg, "frame-opacity", &opts.frame_opacity);
// -z (options.clear_shadow) // -z (clear_shadow)
if (config_lookup_bool(&cfg, "clear-shadow", &ival)) lcfg_lookup_bool(&cfg, "clear-shadow", &opts.clear_shadow);
options.clear_shadow = ival; // -c (shadow_enable)
// -c (options.shadow_enable) if (config_lookup_bool(&cfg, "shadow", &ival) && ival)
if (config_lookup_bool(&cfg, "shadow", &ival) && ival) { wintype_arr_enable(opts.wintype_shadow);
options.shadow_enable = 2; // -C (no_dock_shadow)
wintype_arr_enable(win_type_shadow); lcfg_lookup_bool(&cfg, "no-dock-shadow", &pcfgtmp->no_dock_shadow);
} // -G (no_dnd_shadow)
// -C (options.no_dock_shadow) lcfg_lookup_bool(&cfg, "no-dnd-shadow", &pcfgtmp->no_dnd_shadow);
if (config_lookup_bool(&cfg, "no-dock-shadow", &ival)) // -m (menu_opacity)
options.no_dock_shadow = ival; config_lookup_float(&cfg, "menu-opacity", &pcfgtmp->menu_opacity);
// -G (options.no_dnd_shadow) // -f (fading_enable)
if (config_lookup_bool(&cfg, "no-dnd-shadow", &ival)) if (config_lookup_bool(&cfg, "fading", &ival) && ival)
options.no_dnd_shadow = ival; wintype_arr_enable(opts.wintype_fade);
// -m (options.menu_opacity) // --no-fading-open-close
config_lookup_float(&cfg, "menu-opacity", &options.menu_opacity); lcfg_lookup_bool(&cfg, "no-fading-openclose", &opts.no_fading_openclose);
// -f (options.fading_enable)
if (config_lookup_bool(&cfg, "fading", &ival) && ival) {
options.fading_enable = 2;
wintype_arr_enable(win_type_fade);
}
// --shadow-red // --shadow-red
config_lookup_float(&cfg, "shadow-red", &options.shadow_red); config_lookup_float(&cfg, "shadow-red", &opts.shadow_red);
// --shadow-green // --shadow-green
config_lookup_float(&cfg, "shadow-green", &options.shadow_green); config_lookup_float(&cfg, "shadow-green", &opts.shadow_green);
// --shadow-blue // --shadow-blue
config_lookup_float(&cfg, "shadow-blue", &options.shadow_blue); config_lookup_float(&cfg, "shadow-blue", &opts.shadow_blue);
// --inactive-opacity-override // --inactive-opacity-override
if (config_lookup_bool(&cfg, "inactive-opacity-override", &ival)) lcfg_lookup_bool(&cfg, "inactive-opacity-override",
options.inactive_opacity_override = ival; &opts.inactive_opacity_override);
// --inactive-dim // --inactive-dim
config_lookup_float(&cfg, "inactive-dim", &options.inactive_dim); config_lookup_float(&cfg, "inactive-dim", &opts.inactive_dim);
// --mark-wmwin-focused // --mark-wmwin-focused
if (config_lookup_bool(&cfg, "mark-wmwin-focused", &ival)) lcfg_lookup_bool(&cfg, "mark-wmwin-focused", &opts.mark_wmwin_focused);
options.mark_wmwin_focused = ival; // --mark-ovredir-focused
lcfg_lookup_bool(&cfg, "mark-ovredir-focused",
&opts.mark_ovredir_focused);
// --shadow-exclude // --shadow-exclude
{ {
config_setting_t *setting = config_setting_t *setting =
@ -3151,13 +3166,13 @@ parse_config(char *cpath) {
if (config_setting_is_array(setting)) { if (config_setting_is_array(setting)) {
int i = config_setting_length(setting); int i = config_setting_length(setting);
while (i--) { while (i--) {
condlst_add(&options.shadow_blacklist, condlst_add(&opts.shadow_blacklist,
config_setting_get_string_elem(setting, i)); config_setting_get_string_elem(setting, i));
} }
} }
// Treat it as a single pattern if it's a string // Treat it as a single pattern if it's a string
else if (CONFIG_TYPE_STRING == config_setting_type(setting)) { else if (CONFIG_TYPE_STRING == config_setting_type(setting)) {
condlst_add(&options.shadow_blacklist, condlst_add(&opts.shadow_blacklist,
config_setting_get_string(setting)); config_setting_get_string(setting));
} }
} }
@ -3172,11 +3187,11 @@ parse_config(char *cpath) {
free(str); free(str);
if (setting) { if (setting) {
if (config_setting_lookup_bool(setting, "shadow", &ival)) if (config_setting_lookup_bool(setting, "shadow", &ival))
win_type_shadow[i] = (Bool) ival; opts.wintype_shadow[i] = (Bool) ival;
if (config_setting_lookup_bool(setting, "fade", &ival)) if (config_setting_lookup_bool(setting, "fade", &ival))
win_type_fade[i] = (Bool) ival; opts.wintype_fade[i] = (Bool) ival;
config_setting_lookup_float(setting, "opacity", config_setting_lookup_float(setting, "opacity",
&win_type_opacity[i]); &opts.wintype_opacity[i]);
} }
} }
} }
@ -3200,17 +3215,26 @@ get_cfg(int argc, char *const *argv) {
{ "inactive-dim", required_argument, NULL, 261 }, { "inactive-dim", required_argument, NULL, 261 },
{ "mark-wmwin-focused", no_argument, NULL, 262 }, { "mark-wmwin-focused", no_argument, NULL, 262 },
{ "shadow-exclude", required_argument, NULL, 263 }, { "shadow-exclude", required_argument, NULL, 263 },
{ "mark-ovredir-focused", no_argument, NULL, 264 },
{ "no-fading-openclose", no_argument, NULL, 265 },
// Must terminate with a NULL entry // Must terminate with a NULL entry
{ NULL, 0, NULL, 0 }, { NULL, 0, NULL, 0 },
}; };
struct options_tmp cfgtmp = {
.no_dock_shadow = False,
.no_dnd_shadow = False,
.menu_opacity = 1.0,
};
Bool shadow_enable = False, fading_enable = False;
int o, longopt_idx, i; int o, longopt_idx, i;
char *config_file = NULL; char *config_file = NULL;
char *lc_numeric_old = mstrcpy(setlocale(LC_NUMERIC, NULL));
for (i = 0; i < NUM_WINTYPES; ++i) { for (i = 0; i < NUM_WINTYPES; ++i) {
win_type_fade[i] = False; opts.wintype_fade[i] = False;
win_type_shadow[i] = False; opts.wintype_shadow[i] = False;
win_type_opacity[i] = 1.0; opts.wintype_opacity[i] = 1.0;
} }
// Pre-parse the commandline arguments to check for --config and invalid // Pre-parse the commandline arguments to check for --config and invalid
@ -3224,67 +3248,71 @@ get_cfg(int argc, char *const *argv) {
} }
#ifdef CONFIG_LIBCONFIG #ifdef CONFIG_LIBCONFIG
parse_config(config_file); parse_config(config_file, &cfgtmp);
#endif #endif
// Parse commandline arguments. Range checking will be done later. // Parse commandline arguments. Range checking will be done later.
// Enforce LC_NUMERIC locale "C" here to make sure dots are recognized
// instead of commas in atof().
setlocale(LC_NUMERIC, "C");
optind = 1; optind = 1;
while (-1 != while (-1 !=
(o = getopt_long(argc, argv, shortopts, longopts, &longopt_idx))) { (o = getopt_long(argc, argv, shortopts, longopts, &longopt_idx))) {
switch (o) { switch (o) {
// Short options // Short options
case 'd': case 'd':
options.display = optarg; opts.display = optarg;
break; break;
case 'D': case 'D':
options.fade_delta = atoi(optarg); opts.fade_delta = atoi(optarg);
break; break;
case 'I': case 'I':
options.fade_in_step = normalize_d(atof(optarg)) * OPAQUE; opts.fade_in_step = normalize_d(atof(optarg)) * OPAQUE;
break; break;
case 'O': case 'O':
options.fade_out_step = normalize_d(atof(optarg)) * OPAQUE; opts.fade_out_step = normalize_d(atof(optarg)) * OPAQUE;
break; break;
case 'c': case 'c':
if (2 != options.shadow_enable) shadow_enable = True;
options.shadow_enable = 1;
break; break;
case 'C': case 'C':
options.no_dock_shadow = True; cfgtmp.no_dock_shadow = True;
break;
case 'G':
cfgtmp.no_dnd_shadow = True;
break; break;
case 'm': case 'm':
options.menu_opacity = atof(optarg); cfgtmp.menu_opacity = atof(optarg);
break; break;
case 'f': case 'f':
if (2 != options.fading_enable)
options.fading_enable = 1;
break;
case 'F': case 'F':
options.fade_trans = True; fading_enable = True;
break; break;
case 'S': case 'S':
options.synchronize = True; opts.synchronize = True;
break; break;
case 'r': case 'r':
options.shadow_radius = atoi(optarg); opts.shadow_radius = atoi(optarg);
break; break;
case 'o': case 'o':
options.shadow_opacity = atof(optarg); opts.shadow_opacity = atof(optarg);
break; break;
case 'l': case 'l':
options.shadow_offset_x = atoi(optarg); opts.shadow_offset_x = atoi(optarg);
break; break;
case 't': case 't':
options.shadow_offset_y = atoi(optarg); opts.shadow_offset_y = atoi(optarg);
break; break;
case 'i': case 'i':
options.inactive_opacity = (normalize_d(atof(optarg)) * OPAQUE); opts.inactive_opacity = (normalize_d(atof(optarg)) * OPAQUE);
break; break;
case 'e': case 'e':
options.frame_opacity = atof(optarg); opts.frame_opacity = atof(optarg);
break; break;
case 'z': case 'z':
options.clear_shadow = True; opts.clear_shadow = True;
break; break;
case 'n': case 'n':
case 'a': case 'a':
@ -3292,11 +3320,8 @@ get_cfg(int argc, char *const *argv) {
fprintf(stderr, "Warning: " fprintf(stderr, "Warning: "
"-n, -a, and -s have been removed.\n"); "-n, -a, and -s have been removed.\n");
break; break;
case 'G':
options.no_dnd_shadow = True;
break;
case 'b': case 'b':
options.fork_after_register = True; opts.fork_after_register = True;
break; break;
// Long options // Long options
case 256: case 256:
@ -3304,31 +3329,39 @@ get_cfg(int argc, char *const *argv) {
break; break;
case 257: case 257:
// --shadow-red // --shadow-red
options.shadow_red = atof(optarg); opts.shadow_red = atof(optarg);
break; break;
case 258: case 258:
// --shadow-green // --shadow-green
options.shadow_green = atof(optarg); opts.shadow_green = atof(optarg);
break; break;
case 259: case 259:
// --shadow-blue // --shadow-blue
options.shadow_blue = atof(optarg); opts.shadow_blue = atof(optarg);
break; break;
case 260: case 260:
// --inactive-opacity-override // --inactive-opacity-override
options.inactive_opacity_override = True; opts.inactive_opacity_override = True;
break; break;
case 261: case 261:
// --inactive-dim // --inactive-dim
options.inactive_dim = atof(optarg); opts.inactive_dim = atof(optarg);
break; break;
case 262: case 262:
// --mark-wmwin-focused // --mark-wmwin-focused
options.mark_wmwin_focused = True; opts.mark_wmwin_focused = True;
break; break;
case 263: case 263:
// --shadow-exclude // --shadow-exclude
condlst_add(&options.shadow_blacklist, optarg); condlst_add(&opts.shadow_blacklist, optarg);
break;
case 264:
// --mark-ovredir-focused
opts.mark_ovredir_focused = True;
break;
case 265:
// --no-fading-openclose
opts.no_fading_openclose = True;
break; break;
default: default:
usage(); usage();
@ -3336,44 +3369,47 @@ get_cfg(int argc, char *const *argv) {
} }
} }
// Restore LC_NUMERIC
setlocale(LC_NUMERIC, lc_numeric_old);
free(lc_numeric_old);
// Range checking and option assignments // Range checking and option assignments
options.fade_delta = max_i(options.fade_delta, 1); opts.fade_delta = max_i(opts.fade_delta, 1);
options.shadow_radius = max_i(options.shadow_radius, 1); opts.shadow_radius = max_i(opts.shadow_radius, 1);
options.shadow_red = normalize_d(options.shadow_red); opts.shadow_red = normalize_d(opts.shadow_red);
options.shadow_green = normalize_d(options.shadow_green); opts.shadow_green = normalize_d(opts.shadow_green);
options.shadow_blue = normalize_d(options.shadow_blue); opts.shadow_blue = normalize_d(opts.shadow_blue);
options.inactive_dim = normalize_d(options.inactive_dim); opts.inactive_dim = normalize_d(opts.inactive_dim);
options.frame_opacity = normalize_d(options.frame_opacity); opts.frame_opacity = normalize_d(opts.frame_opacity);
options.shadow_opacity = normalize_d(options.shadow_opacity); opts.shadow_opacity = normalize_d(opts.shadow_opacity);
options.menu_opacity = normalize_d(options.menu_opacity); cfgtmp.menu_opacity = normalize_d(cfgtmp.menu_opacity);
if (OPAQUE == options.inactive_opacity) { if (OPAQUE == opts.inactive_opacity) {
options.inactive_opacity = 0; opts.inactive_opacity = 0;
} }
if (1 == options.shadow_enable) if (shadow_enable)
wintype_arr_enable(win_type_shadow); wintype_arr_enable(opts.wintype_shadow);
win_type_shadow[WINTYPE_DESKTOP] = False; opts.wintype_shadow[WINTYPE_DESKTOP] = False;
if (options.no_dock_shadow) if (cfgtmp.no_dock_shadow)
win_type_shadow[WINTYPE_DOCK] = False; opts.wintype_shadow[WINTYPE_DOCK] = False;
if (options.no_dnd_shadow) if (cfgtmp.no_dnd_shadow)
win_type_shadow[WINTYPE_DND] = False; opts.wintype_shadow[WINTYPE_DND] = False;
if (1 == options.fading_enable) { if (fading_enable)
wintype_arr_enable(win_type_fade); wintype_arr_enable(opts.wintype_fade);
} if (1.0 != cfgtmp.menu_opacity) {
if (1.0 != options.menu_opacity) { opts.wintype_opacity[WINTYPE_DROPDOWN_MENU] = cfgtmp.menu_opacity;
win_type_opacity[WINTYPE_DROPDOWN_MENU] = options.menu_opacity; opts.wintype_opacity[WINTYPE_POPUP_MENU] = cfgtmp.menu_opacity;
win_type_opacity[WINTYPE_POPUP_MENU] = options.menu_opacity;
} }
// Other variables determined by options // Other variables determined by options
// Determine whether we need to track focus changes // Determine whether we need to track focus changes
if (options.inactive_opacity || options.inactive_dim) { if (opts.inactive_opacity || opts.inactive_dim) {
options.track_focus = True; opts.track_focus = True;
} }
// Determine whether we need to track window name and class // Determine whether we need to track window name and class
if (options.shadow_blacklist || options.fade_blacklist) if (opts.shadow_blacklist || opts.fade_blacklist)
options.track_wdata = True; opts.track_wdata = True;
} }
static void static void
@ -3441,14 +3477,14 @@ main(int argc, char **argv) {
fade_time = get_time_in_milliseconds(); fade_time = get_time_in_milliseconds();
dpy = XOpenDisplay(options.display); dpy = XOpenDisplay(opts.display);
if (!dpy) { if (!dpy) {
fprintf(stderr, "Can't open display\n"); fprintf(stderr, "Can't open display\n");
exit(1); exit(1);
} }
XSetErrorHandler(error); XSetErrorHandler(error);
if (options.synchronize) { if (opts.synchronize) {
XSynchronize(dpy, 1); XSynchronize(dpy, 1);
} }
@ -3490,13 +3526,13 @@ main(int argc, char **argv) {
register_cm(scr); register_cm(scr);
if (options.fork_after_register) fork_after(); if (opts.fork_after_register) fork_after();
get_atoms(); get_atoms();
pa.subwindow_mode = IncludeInferiors; pa.subwindow_mode = IncludeInferiors;
gaussian_map = make_gaussian_map(dpy, options.shadow_radius); gaussian_map = make_gaussian_map(dpy, opts.shadow_radius);
presum_gaussian(gaussian_map); presum_gaussian(gaussian_map);
root_width = DisplayWidth(dpy, scr); root_width = DisplayWidth(dpy, scr);
@ -3510,16 +3546,16 @@ main(int argc, char **argv) {
// Generates another Picture for shadows if the color is modified by // Generates another Picture for shadows if the color is modified by
// user // user
if (!options.shadow_red && !options.shadow_green && !options.shadow_blue) { if (!opts.shadow_red && !opts.shadow_green && !opts.shadow_blue) {
cshadow_picture = black_picture; cshadow_picture = black_picture;
} else { } else {
cshadow_picture = solid_picture(dpy, True, 1, cshadow_picture = solid_picture(dpy, True, 1,
options.shadow_red, options.shadow_green, options.shadow_blue); opts.shadow_red, opts.shadow_green, opts.shadow_blue);
} }
// Generates a picture for options.inactive_dim // Generates a picture for inactive_dim
if (options.inactive_dim) { if (opts.inactive_dim) {
dim_picture = solid_picture(dpy, True, options.inactive_dim, 0, 0, 0); dim_picture = solid_picture(dpy, True, opts.inactive_dim, 0, 0, 0);
} }
all_damage = None; all_damage = None;
@ -3543,7 +3579,7 @@ main(int argc, char **argv) {
XFree(children); XFree(children);
if (options.track_focus) { if (opts.track_focus) {
recheck_focus(dpy); recheck_focus(dpy);
} }

View File

@ -242,21 +242,40 @@ typedef struct _win {
} win; } win;
typedef struct _options { typedef struct _options {
// General
char *display; char *display;
int shadow_radius; /// Whether to try to detect WM windows and mark them as focused.
int shadow_offset_x; Bool mark_wmwin_focused;
int shadow_offset_y; /// Whether to mark override-redirect windows as focused.
double shadow_opacity; Bool mark_ovredir_focused;
/// Whether to fork to background.
Bool fork_after_register;
Bool synchronize;
// Shadow
Bool wintype_shadow[NUM_WINTYPES];
/// Red, green and blue tone of the shadow.
double shadow_red, shadow_green, shadow_blue;
int shadow_radius;
int shadow_offset_x, shadow_offset_y;
double shadow_opacity;
Bool clear_shadow;
/// Shadow blacklist. A linked list of conditions.
wincond *shadow_blacklist;
// Fading
Bool wintype_fade[NUM_WINTYPES];
/// How much to fade in in a single fading step. /// How much to fade in in a single fading step.
opacity_t fade_in_step; opacity_t fade_in_step;
/// How much to fade out in a single fading step. /// How much to fade out in a single fading step.
opacity_t fade_out_step; opacity_t fade_out_step;
unsigned long fade_delta; unsigned long fade_delta;
Bool fade_trans; Bool no_fading_openclose;
/// Fading blacklist. A linked list of conditions.
Bool clear_shadow; wincond *fade_blacklist;
// Opacity
double wintype_opacity[NUM_WINTYPES];
/// Default opacity for inactive windows. /// Default opacity for inactive windows.
/// 32-bit integer with the format of _NET_WM_OPACITY. 0 stands for /// 32-bit integer with the format of _NET_WM_OPACITY. 0 stands for
/// not enabled, default. /// not enabled, default.
@ -268,35 +287,19 @@ typedef struct _options {
/// How much to dim an inactive window. 0.0 - 1.0, 0 to disable. /// How much to dim an inactive window. 0.0 - 1.0, 0 to disable.
double inactive_dim; double inactive_dim;
/// Whether to try to detect WM windows and mark them as focused. // Calculated
double mark_wmwin_focused;
/// Whether compton needs to track focus changes. /// Whether compton needs to track focus changes.
Bool track_focus; Bool track_focus;
/// Whether compton needs to track window name and class. /// Whether compton needs to track window name and class.
Bool track_wdata; Bool track_wdata;
/// Shadow blacklist. A linked list of conditions. } options_t;
wincond *shadow_blacklist;
/// Fading blacklist. A linked list of conditions.
wincond *fade_blacklist;
/// Whether to fork to background. struct options_tmp {
Bool fork_after_register;
/// Red, green and blue tone of the shadow.
double shadow_red;
double shadow_green;
double shadow_blue;
Bool synchronize;
// Temporary options
int shadow_enable;
int fading_enable;
Bool no_dock_shadow; Bool no_dock_shadow;
Bool no_dnd_shadow; Bool no_dnd_shadow;
double menu_opacity; double menu_opacity;
} options_t; };
typedef struct _conv { typedef struct _conv {
int size; int size;
@ -722,6 +725,9 @@ map_win(Display *dpy, Window id,
unsigned long sequence, Bool fade, unsigned long sequence, Bool fade,
Bool override_redirect); Bool override_redirect);
static void
finish_map_win(Display *dpy, win *w);
static void static void
finish_unmap_win(Display *dpy, win *w); finish_unmap_win(Display *dpy, win *w);
@ -917,11 +923,19 @@ static void
fork_after(void); fork_after(void);
#ifdef CONFIG_LIBCONFIG #ifdef CONFIG_LIBCONFIG
static void
lcfg_lookup_bool(const config_t *config, const char *path, Bool *value) {
int ival;
if (config_lookup_bool(config, path, &ival))
*value = ival;
}
static FILE * static FILE *
open_config_file(char *cpath, char **path); open_config_file(char *cpath, char **path);
static void static void
parse_config(char *cpath); parse_config(char *cpath, struct options_tmp *pcfgtmp);
#endif #endif
static void static void