diff options
| author | Vito Caputo <vcaputo@gnugeneration.com> | 2015-02-21 14:37:46 -0800 | 
|---|---|---|
| committer | Vito Caputo <vcaputo@gnugeneration.com> | 2015-02-21 14:43:20 -0800 | 
| commit | a8dbe9a9ad1f31134ed55bb20da21ccd6eb004eb (patch) | |
| tree | e2222dac84e96934f097a8fd307a00d1c3247c75 | |
| parent | d0386f4dc9fcc1fdbc490b586732cbb9d5f935bf (diff) | |
Introduced a 'refocus original window' operation
See changes to README, but basically if you're holding down Alt and doing
some window management operations, while still pressing Alt hit the other
Alt key to return focus to the starting window+desktop.  Nothing is undone,
focus is simply returned to the starting window+desktop.
| -rw-r--r-- | README | 11 | ||||
| -rw-r--r-- | TODO | 19 | ||||
| -rw-r--r-- | vwm.c | 47 | 
3 files changed, 63 insertions, 14 deletions
| @@ -119,6 +119,17 @@ Built-ins:        Think of the Mod1 release as a transaction commit when coupled with        the *'d commands. +      If a simultaneous second Mod1 is pressed at any point during a *'d +      command, the window (and its desktop) focused when the *'d command +      began will immediately be refocused - but not raised.  This is +      intentional to simplify the arranging of obscured focused windows.  If +      you find yourself restored to a desktop full of windows where your +      focused window is totally obscured/invisible, simply press Mod1-k to +      raise it if desired. + +      At any point during a *'d operation one may (re)press a second Mod1 +      to return to the origin, it is not limited to a single use. +  Default launchers (configure by editing launchers.def and rebuild):    Mod1-x              xterm @@ -140,6 +140,25 @@ XINERAMA:  BUGS: +- The introduction of re-Mod1 origin restoration has made some inconsistencies +  much more visible; if you enter the shelf and try re-Mod1 to your origin, +  nothing happens because shelf entry doesn't grab the keyboard / start a +  "transaction".  It's probably time to revisit the categorizing of which +  operations are grabbed vs. ungrabbed, because the utility of re-Mod1 is such +  that unless it always does what you expect it's very unnerving/jarring, +  because you become dependent on its remembering where you started and when +  for whatever reason it didn't remember your origin there's a WTF GRR moment, +  relative to how awesome it is normally. + +  Also the current implementation returns you to the origin *window*, if your +  operation migrated the window and you try to return to origin, you go nowhere +  because you brought the window with.  In this situation, you probably expected +  it to return you to your origin *desktop*, and it should probably return you +  there when the operation was a window migrate, for example. + +  Let's live with the minimal changes for now and see where all the UX rough +  edges are before going too crazy. +  - The halfscreen/quarterscreen shortcuts are treated as user-configured windows    clobbering the cached original client configuration, I don't think this is    the right thing to do.  I'm leaning towards all the shortcutted @@ -1,7 +1,7 @@  /*   *                                  \/\/\   * - *  Copyright (C) 2012-2014  Vito Caputo - <vcaputo@gnugeneration.com> + *  Copyright (C) 2012-2015  Vito Caputo - <vcaputo@gnugeneration.com>   *   *  This program is free software: you can redistribute it and/or modify it   *  under the terms of the GNU General Public License version 3 as published @@ -91,6 +91,7 @@ static LIST_HEAD(desktops_mru);							/* global list of all (virtual) desktops i  static LIST_HEAD(windows_mru);							/* global list of all managed windows kept in MRU order */  static LIST_HEAD(xwindows);							/* global list of all xwindows kept in the X server stacking order */  static vwm_window_t		*console = NULL;				/* the console window */ +static vwm_window_t		*focused_origin = NULL;				/* the originating window in a grabbed operation/transaction */  static vwm_desktop_t		*focused_desktop = NULL;			/* currently focused (virtual) desktop */  static vwm_window_t		*focused_shelf = NULL;				/* currently focused shelved window */  static vwm_context_focus_t	focused_context = VWM_CONTEXT_FOCUS_DESKTOP;	/* currently focused context */  @@ -1877,6 +1878,7 @@ static vwm_xwindow_t * vwm_win_unmanage(vwm_window_t *vwin)  	list_del(&vwin->windows_mru);  	if(vwin == console) console = NULL; +	if(vwin == focused_origin) focused_origin = NULL;  	vwin->xwindow->managed = NULL; @@ -2423,6 +2425,14 @@ static void vwm_keyreleased(Window win, XEvent *keyrelease)  		case XK_Alt_R:  		case XK_Alt_L:	/* TODO: actually use the modifier mapping, for me XK_Alt_[LR] is Mod1.  XGetModifierMapping()... */  			VWM_TRACE("XK_Alt_[LR] released"); + +			/* aborted? try restore focused_origin */ +			if(key_is_grabbed > 1 && focused_origin) { +				VWM_TRACE("restoring %p on %p", focused_origin, focused_origin->desktop); +				vwm_desktop_focus(focused_origin->desktop); +				vwm_win_focus(focused_origin); +			} +  			/* make the focused window the most recently used */  			if((vwin = vwm_win_focused())) vwm_win_mru(vwin); @@ -2464,6 +2474,8 @@ static void vwm_keypressed(Window win, XEvent *keypress)  		repeat_cnt = 0;  	} +	vwin = vwm_win_focused(); +  	switch(sym) {  #define launcher(_sym, _label, _argv)\ @@ -2475,6 +2487,11 @@ static void vwm_keypressed(Window win, XEvent *keypress)  		}  #include "launchers.def"  #undef launcher +		case XK_Alt_L: /* transaction abort */ +		case XK_Alt_R: +			if(key_is_grabbed) key_is_grabbed++; +			VWM_TRACE("aborting with origin %p", focused_origin); +			break;  		case XK_grave: /* toggle shelf visibility */  			vwm_context_focus(VWM_CONTEXT_FOCUS_OTHER); @@ -2484,7 +2501,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			do_grab = 1; /* update MRU window on commit (Mod1 release) */  			/* focus the next window, note this doesn't affect MRU yet, that happens on Mod1 release */ -			if((vwin = vwm_win_focused())) { +			if(vwin) {  				if(keypress->xkey.state & ShiftMask) {  					vwm_win_focus_next(vwin, focused_context, VWM_FENCE_MASKED_VIOLATE);  				} else { @@ -2500,7 +2517,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			if(keypress->xkey.state & ShiftMask) {  				/* migrate the focused window with the desktop focus to the most recently used desktop */ -				if((vwin = vwm_win_focused())) vwm_win_migrate(vwin, next_desktop); +				if(vwin) vwm_win_migrate(vwin, next_desktop);  			} else {  				vwm_desktop_focus(next_desktop);  			} @@ -2508,7 +2525,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  		}  		case XK_d: /* destroy focused */ -			if((vwin = vwm_win_focused())) { +			if(vwin) {  				if(keypress->xkey.state & ShiftMask) {  /* brutally destroy the focused window */  					XKillClient(display, vwin->xwindow->id);  				} else { /* kindly destroy the focused window */ @@ -2533,7 +2550,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			do_grab = 1; /* update MRU desktop on commit (Mod1 release) */  			if(keypress->xkey.state & ShiftMask) { -				if((vwin = vwm_win_focused())) { +				if(vwin) {  					/* migrate the focused window to a newly created virtual desktop, focusing the new desktop simultaneously */  					vwm_win_migrate(vwin, vwm_desktop_create(NULL));  				} @@ -2547,7 +2564,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			do_grab = 1; /* update MRU desktop on commit (Mod1 release) */  			if(keypress->xkey.state & ShiftMask) { -				if((vwin = vwm_win_focused()) && vwin->desktop->desktops.prev != &desktops) { +				if(vwin && vwin->desktop->desktops.prev != &desktops) {  					/* migrate the focused window with the desktop focus to the previous desktop */  					vwm_win_migrate(vwin, list_entry(vwin->desktop->desktops.prev, vwm_desktop_t, desktops));  				} @@ -2566,7 +2583,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			do_grab = 1; /* update MRU desktop on commit (Mod1 release) */  			if(keypress->xkey.state & ShiftMask) { -				if((vwin = vwm_win_focused()) && vwin->desktop->desktops.next != &desktops) { +				if(vwin && vwin->desktop->desktops.next != &desktops) {  					/* migrate the focused window with the desktop focus to the next desktop */  					vwm_win_migrate(vwin, list_entry(vwin->desktop->desktops.next, vwm_desktop_t, desktops));  				} @@ -2582,7 +2599,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			break;  		case XK_k: /* raise or shelve the focused window */ -			if((vwin = vwm_win_focused())) { +			if(vwin) {  				if(keypress->xkey.state & ShiftMask) { /* shelf the window and focus the shelf */  					if(focused_context != VWM_CONTEXT_FOCUS_SHELF) {  						/* shelve the focused window while focusing the shelf */ @@ -2615,7 +2632,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			break;  		case XK_j: /* lower or unshelve the focused window */ -			if((vwin = vwm_win_focused())) { +			if(vwin) {  				if(keypress->xkey.state & ShiftMask) { /* unshelf the window to the focused desktop, and focus the desktop */  					if(focused_context == VWM_CONTEXT_FOCUS_SHELF) {  						/* unshelve the focused window, focus the desktop it went to */ @@ -2633,7 +2650,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			break;  		case XK_Return: /* (full-screen / restore) focused window */ -			if((vwin = vwm_win_focused())) { +			if(vwin) {  				if(vwin->autoconfigured) {  					vwm_win_autoconf(vwin, VWM_SCREEN_REL_XWIN, VWM_WIN_AUTOCONF_NONE);  				} else { @@ -2643,11 +2660,11 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			break;  		case XK_s: /* shelve focused window */ -			if((vwin = vwm_win_focused()) && !vwin->shelved) vwm_win_shelve(vwin); +			if(vwin && !vwin->shelved) vwm_win_shelve(vwin);  			break;  		case XK_bracketleft:	/* reconfigure the focused window to occupy the left or top half of the screen or left quarters on repeat */ -			if((vwin = vwm_win_focused())) { +			if(vwin) {  				do_grab = 1;  				if(keypress->xkey.state & ShiftMask) { @@ -2667,7 +2684,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			break;  		case XK_bracketright:	/* reconfigure the focused window to occupy the right or bottom half of the screen or right quarters on repeat */ -			if((vwin = vwm_win_focused())) { +			if(vwin) {  				do_grab = 1;  				if(keypress->xkey.state & ShiftMask) { @@ -2691,7 +2708,7 @@ static void vwm_keypressed(Window win, XEvent *keypress)  			break;  		case XK_apostrophe:	/* reset snowflakes of the focused window, suppressed when not compositing */ -			if((vwin = vwm_win_focused()) && compositing_mode && vwin->xwindow->overlay.snowflakes_cnt) { +			if(vwin && compositing_mode && vwin->xwindow->overlay.snowflakes_cnt) {  				vwin->xwindow->overlay.snowflakes_cnt = 0;  				vwm_comp_damage_win(vwin->xwindow);  			} @@ -2711,6 +2728,8 @@ static void vwm_keypressed(Window win, XEvent *keypress)  	/* if what we're doing requests a grab, if not already grabbed, grab keyboard */  	if(!key_is_grabbed && do_grab) { +		VWM_TRACE("saving focused_origin of %p", vwin); +		focused_origin = vwin; /* for returning to on abort */  		XGrabKeyboard(display, RootWindow(display, screen_num), False, GrabModeAsync, GrabModeAsync, CurrentTime);  		key_is_grabbed = 1;  	} | 
