aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--XMonad/Actions/GroupNavigation.hs210
-rw-r--r--xmonad-contrib.cabal1
2 files changed, 211 insertions, 0 deletions
diff --git a/XMonad/Actions/GroupNavigation.hs b/XMonad/Actions/GroupNavigation.hs
new file mode 100644
index 0000000..adde3f9
--- /dev/null
+++ b/XMonad/Actions/GroupNavigation.hs
@@ -0,0 +1,210 @@
+{-# LANGUAGE DeriveDataTypeable #-}
+
+----------------------------------------------------------------------
+-- |
+-- Module : XMonad.Actions.GroupNavigation
+-- Copyright : (c) nzeh@cs.dal.ca
+-- License : BSD3-style (see LICENSE)
+--
+-- Maintainer : nzeh@cs.dal.ca
+-- Stability : unstable
+-- Portability : unportable
+--
+-- Provides methods for cycling through groups of windows across
+-- workspaces, ignoring windows that do not belong to this group. A
+-- group consists of all windows matching a user-provided boolean
+-- query.
+--
+-- Also provides a method for jumping back to the most recently used
+-- window in any given group.
+--
+----------------------------------------------------------------------
+
+module XMonad.Actions.GroupNavigation ( -- * Usage
+ -- $usage
+ Direction (..)
+ , nextMatch
+ , nextMatchOrDo
+ , nextMatchWithThis
+ , historyHook
+ ) where
+
+import Control.Monad
+import Control.Monad.Reader
+import Data.Foldable
+import Data.Map as Map
+import Data.Maybe
+import Data.Sequence as Seq
+import Data.Set as Set
+import Graphics.X11.Types
+import Prelude hiding (concatMap, drop, elem, filter, null, reverse)
+import XMonad.Core
+import XMonad.ManageHook
+import XMonad.Operations
+import qualified XMonad.StackSet as SS
+import qualified XMonad.Util.ExtensibleState as XS
+
+{- $usage
+
+Import the module into your @~\/.xmonad\/xmonad.hs@:
+
+> import XMonad.Actions,GroupNavigation
+
+To support cycling forward and backward through all xterm windows, add
+something like this to your keybindings:
+
+> , ((modm , xK_t), nextMatch Forward (className =? "XTerm"))
+> , ((modm .|. shiftMask, xK_t), nextMatch Backward (className =? "XTerm"))
+
+These key combinations do nothing if there is no xterm window open.
+If you rather want to open a new xterm window if there is no open
+xterm window, use 'nextMatchOrDo' instead:
+
+> , ((modm , xK_t), nextMatchOrDo Forward (className =? "XTerm") (spawn "xterm"))
+> , ((modm .|. shiftMask, xK_t), nextMatchOrDo Backward (className =? "XTerm") (spawn "xterm"))
+
+You can use 'nextMatchWithThis' with an arbitrary query to cycle
+through all windows for which this query returns the same value as the
+current window. For example, to cycle through all windows in the same
+window class as the current window use:
+
+> , ((modm , xK_f), nextMatchWithThis Forward className)
+> , ((modm , xK_b), nextMatchWithThis Backward className)
+
+Finally, you can define keybindings to jump to the most recent window
+matching a certain Boolean query. To do this, you need to add
+'historyHook' to your logHook:
+
+> main = xmonad $ defaultConfig { logHook = historyHook }
+
+Then the following keybindings, for example, allow you to return to
+the most recent xterm or emacs window or to simply to the most recent
+window:
+
+> , ((modm .|. controlMask, xK_e), nextMatch History (className =? "Emacs"))
+> , ((modm .|. controlMask, xK_t), nextMatch History (className =? "XTerm"))
+> , ((modm , xK_BackSpace), nextMatch History (return True))
+
+Again, you can use 'nextMatchOrDo' instead of 'nextMatch' if you want
+to execute an action if no window matching the query exists. -}
+
+--- Basic cyclic navigation based on queries -------------------------
+
+-- | The direction in which to look for the next match
+data Direction = Forward -- ^ Forward from current window or workspace
+ | Backward -- ^ Backward from current window or workspace
+ | History -- ^ Backward in history
+
+-- | Focuses the next window for which the given query produces the
+-- same result as the currently focused window. Does nothing if there
+-- is no focused window (i.e., the current workspace is empty).
+nextMatchWithThis :: Eq a => Direction -> Query a -> X ()
+nextMatchWithThis dir qry = withFocused $ \win -> do
+ prop <- runQuery qry win
+ nextMatch dir (qry =? prop)
+
+-- | Focuses the next window that matches the given boolean query.
+-- Does nothing if there is no such window. This is the same as
+-- 'nextMatchOrDo' with alternate action @return ()@.
+nextMatch :: Direction -> Query Bool -> X ()
+nextMatch dir qry = nextMatchOrDo dir qry (return ())
+
+-- | Focuses the next window that matches the given boolean query. If
+-- there is no such window, perform the given action instead.
+nextMatchOrDo :: Direction -> Query Bool -> X () -> X ()
+nextMatchOrDo dir qry act = orderedWindowList dir
+ >>= focusNextMatchOrDo qry act
+
+-- Produces the action to perform depending on whether there's a
+-- matching window
+focusNextMatchOrDo :: Query Bool -> X () -> Seq Window -> X ()
+focusNextMatchOrDo qry act = findM (runQuery qry)
+ >=> maybe act (windows . SS.focusWindow)
+
+-- Returns the list of windows ordered by workspace as specified in
+-- ~/.xmonad/xmonad.hs
+orderedWindowList :: Direction -> X (Seq Window)
+orderedWindowList History = liftM (\(HistoryDB w ws) -> maybe ws (ws |>) w) XS.get
+orderedWindowList dir = withWindowSet $ \ss -> do
+ wsids <- asks (Seq.fromList . workspaces . config)
+ let wspcs = orderedWorkspaceList ss wsids
+ wins = dirfun dir
+ $ foldl' (><) Seq.empty
+ $ fmap (Seq.fromList . SS.integrate' . SS.stack) wspcs
+ cur = currentWindow ss
+ return $ maybe wins (rotfun wins) cur
+ where
+ dirfun Backward = Seq.reverse
+ dirfun _ = id
+ rotfun wins x = rotate $ rotateTo (== x) wins
+
+-- Returns the currently focused window or Nothing if no window is
+-- currently focused.
+currentWindow :: WindowSet -> Maybe Window
+currentWindow = liftM SS.focus . SS.stack . SS.workspace . SS.current
+
+-- Returns the ordered workspace list as specified in ~/.xmonad/xmonad.hs
+orderedWorkspaceList :: WindowSet -> Seq String -> Seq WindowSpace
+orderedWorkspaceList ss wsids = rotateTo isCurWS wspcs'
+ where
+ wspcs = SS.workspaces ss
+ wspcsMap = foldl' (\m ws -> Map.insert (SS.tag ws) ws m) Map.empty wspcs
+ wspcs' = fmap (\wsid -> wspcsMap ! wsid) wsids
+ isCurWS ws = SS.tag ws == SS.tag (SS.workspace $ SS.current ss)
+
+--- History navigation, requires a layout modifier -------------------
+
+-- The state extension that holds the history information
+data HistoryDB = HistoryDB (Maybe Window) -- currently focused window
+ (Seq Window) -- previously focused windows
+ deriving (Read, Show, Typeable)
+
+instance ExtensionClass HistoryDB where
+
+ initialValue = HistoryDB Nothing Seq.empty
+ extensionType = PersistentExtension
+
+-- | Action that needs to be executed as a logHook to maintain the
+-- focus history of all windows as the WindowSet changes.
+historyHook :: X ()
+historyHook = XS.get >>= updateHistory >>= XS.put
+
+-- Updates the history in response to a WindowSet change
+updateHistory :: HistoryDB -> X HistoryDB
+updateHistory (HistoryDB oldcur oldhist) = withWindowSet $ \ss -> do
+ let newcur = currentWindow ss
+ wins = Set.fromList $ SS.allWindows ss
+ newhist = Seq.filter (flip Set.member wins) oldhist
+ return $ HistoryDB newcur (del newcur $ ins oldcur newhist)
+ where
+ ins x xs = maybe xs (<| xs) x
+ del x xs = maybe xs (\x' -> Seq.filter (/= x') xs) x
+
+--- Some sequence helpers --------------------------------------------
+
+-- Rotates the sequence by one position
+rotate :: Seq a -> Seq a
+rotate xs = rotate' (viewl xs)
+ where
+ rotate' EmptyL = Seq.empty
+ rotate' (x' :< xs') = xs' |> x'
+
+-- Rotates the sequence until an element matching the given condition
+-- is at the beginning of the sequence.
+rotateTo :: (a -> Bool) -> Seq a -> Seq a
+rotateTo cond xs = let (lxs, rxs) = breakl cond xs in rxs >< lxs
+
+--- A monadic find ---------------------------------------------------
+
+-- Applies the given action to every sequence element in turn until
+-- the first element is found for which the action returns true. The
+-- remaining elements in the sequence are ignored.
+findM :: Monad m => (a -> m Bool) -> Seq a -> m (Maybe a)
+findM cond xs = findM' cond (viewl xs)
+ where
+ findM' _ EmptyL = return Nothing
+ findM' qry (x' :< xs') = do
+ isMatch <- qry x'
+ if isMatch
+ then return (Just x')
+ else findM qry xs'
diff --git a/xmonad-contrib.cabal b/xmonad-contrib.cabal
index 91518d4..59a252a 100644
--- a/xmonad-contrib.cabal
+++ b/xmonad-contrib.cabal
@@ -96,6 +96,7 @@ library
XMonad.Actions.FloatSnap
XMonad.Actions.FocusNth
XMonad.Actions.GridSelect
+ XMonad.Actions.GroupNavigation
XMonad.Actions.MessageFeedback
XMonad.Actions.MouseGestures
XMonad.Actions.MouseResize