aboutsummaryrefslogtreecommitdiffstats
path: root/XMonad/Actions
diff options
context:
space:
mode:
Diffstat (limited to 'XMonad/Actions')
-rw-r--r--XMonad/Actions/WorkspaceCursors.hs193
1 files changed, 121 insertions, 72 deletions
diff --git a/XMonad/Actions/WorkspaceCursors.hs b/XMonad/Actions/WorkspaceCursors.hs
index 9aedc5f..631c77d 100644
--- a/XMonad/Actions/WorkspaceCursors.hs
+++ b/XMonad/Actions/WorkspaceCursors.hs
@@ -9,77 +9,108 @@
-- Stability : unstable
-- Portability : portable
--
--- Generalizes plane to arbitrary dimensions.
+-- Like Plane for an arbitrary number of dimensions.
-----------------------------------------------------------------------------
module XMonad.Actions.WorkspaceCursors
(
-- * Usage
-- $usage
- toList
- ,focusDepth
+ focusDepth
+ ,makeCursors
+ ,toList
,workspaceCursors
+ ,WorkspaceCursors
+ ,getFocus
+
+ -- * Modifying the focus
,modifyLayer
- ,makeCursors
- ,sampleCursors
+ ,modifyLayer'
+ ,shiftModifyLayer,shiftLayer
-- * Functions to pass to 'modifyLayer'
,focusNth'
,noWrapUp,noWrapDown
+
+ -- * Todo
+ -- $todo
) where
+import qualified XMonad.StackSet as W
import XMonad.Actions.FocusNth(focusNth')
-import XMonad.Layout.LayoutModifier(LayoutModifier(handleMess,
- redoLayout))
+import XMonad.Layout.LayoutModifier(ModifiedLayout(..),
+ LayoutModifier(handleMess, redoLayout))
import XMonad(Typeable, Message, WorkspaceId, X, XState(windowset),
fromMessage, sendMessage, windows, gets)
+import Control.Monad((<=<), guard, liftM, liftM2, when)
import Control.Applicative((<$>))
-import Control.Monad(Monad(return, (>>=), (>>)), Functor(..),
- guard, unless)
import Data.Foldable(Foldable(foldMap), toList)
-import Data.Maybe(Maybe(Nothing), fromJust, listToMaybe)
-import Data.Monoid(Monoid(mconcat, mappend))
+import Data.Maybe(fromJust, listToMaybe)
+import Data.Monoid(Monoid(mappend, mconcat))
import Data.Traversable(sequenceA)
-import qualified XMonad.StackSet as W
-
-- $usage
--
-- Here is an example config:
--
-- > import XMonad
-- > import XMonad.Actions.WorkspaceCursors
--- > import XMonad.Config.Desktops
+-- > import XMonad.Hooks.DynamicLog
-- > import XMonad.Util.EZConfig
+-- > import qualified XMonad.StackSet as W
-- >
-- > main = do
--- > xmonad $ additionalKeysP desktopConfig
--- > { workspaces = toList sampleCurs
--- > , layoutHook = workspaceCursors myCursors $ layoutHook desktopConfig
--- > }
--- > [("M-"++shift++[k], modifyLayer f depth)
--- > | (f,shift) <- zip [W.focusUp',W.focusDown'] [[],"S-"]
--- > , (depth,k) <- zip [1..focusDepth myCursors] "asdf"]
+-- > x <- xmobar conf
+-- > xmonad x
+-- >
+-- > conf = additionalKeysP defaultConfig
+-- > { layoutHook = workspaceCursors myCursors $ layoutHook defaultConfig
+-- > , workspaces = toList myCursors } $
+-- > [("M-"++shift++control++[k], f direction depth)
+-- > | (f,shift) <- zip [modifyLayer,shiftModifyLayer] ["","S-"]
+-- > , (direction,control) <- zip [W.focusUp',W.focusDown'] ["C-",""]
+-- > , (depth,k) <- zip (reverse [1..focusDepth myCursors]) "asdf"]
+-- > ++ moreKeybindings
+-- >
+-- > moreKeybindings = []
-- >
--- > myCursors = makeCursors $ map (map show) [[1..3],[1..3],[1..3],[1..9]]
+-- > myCursors = makeCursors $ map (map (\x -> [x])) [ "1234", "abc", "xyz"]
+-- > -- myCursors = makeCursors [["wsA","wsB","wsC"],["-alpha-","-beta-","-gamma-"],["x","y"]]
--------------------------------------------------------------------------------
-sampleCursors :: Cursors String
-sampleCursors = makeCursors $ map (map show) [[1..3::Int],[1..3],[1..9]]
+-- $todo
+--
+-- * Find and document how to raise the allowable length of arguments:
+-- restoring xmonad's state results in: @xmonad: executeFile: resource
+-- exhausted (Argument list too long)@ when you specify more than about 50
+-- workspaces. Or change it such that workspaces are created when you try to
+-- view it.
+--
+-- * Function for pretty printing for DynamicLog that groups workspaces by
+-- common prefixes
+--
+-- * Examples of adding workspaces to the cursors, having them appear multiple
+-- times for being able to show jumping to some n'th multiple workspace
+-- | makeCursors requires a nonempty string, and each sublist must be nonempty
makeCursors :: [[String]] -> Cursors String
-makeCursors (x:xs) = Prelude.foldr addDim (end x) xs
-makeCursors [] = error "Cursors cannot be empty"
-
-addDim :: (Monoid a) => [a] -> Cursors a -> Cursors a
-addDim prefixes prev = Cons . fromJust . W.differentiate
- $ map (\p -> fmap (p `mappend`) prev) prefixes
-
-end :: [String] -> Cursors String
+makeCursors [] = error "Workspace Cursors cannot be empty"
+makeCursors a = concat . reverse <$> foldl addDim x xs
+ where x = end $ map return $ head a
+ xs = map (map return) $ tail a
+ -- this could probably be simplified, but this true:
+ -- toList . makeCursors == map (concat . reverse) . sequence . reverse . map (map (:[]))
+ -- the strange order is used because it makes the regular M-1..9
+ -- bindings change the prefixes first
+
+addDim :: (Monoid a) => Cursors a -> [a] -> Cursors a
+addDim prev prefixes = Cons . fromJust . W.differentiate
+ $ map ((<$> prev) . mappend) prefixes
+
+end :: [a] -> Cursors a
end = Cons . fromJust . W.differentiate . map End
data Cursors a
@@ -88,14 +119,17 @@ data Cursors a
instance Foldable Cursors where
foldMap f (End x) = f x
- foldMap f (Cons st) = mconcat $ map (foldMap f) $ W.integrate st
+ foldMap f (Cons (W.Stack x y z)) = foldMap f x `mappend` mconcat (map (foldMap f) $ reverse y ++ z)
instance Functor Cursors where
fmap f (End a) = End $ f a
fmap f (Cons (W.Stack x y z)) = Cons $ W.Stack (fmap f x) (fmap (fmap f) y) (fmap (fmap f) z)
changeFocus :: (Cursors t -> Bool) -> Cursors t -> [Cursors t]
-changeFocus p (Cons x) = chFocus p x >>= changeFocus p . Cons
+changeFocus p (Cons x) = do
+ choose <- chFocus p x
+ foc <- changeFocus p $ W.focus choose
+ return . Cons $ choose { W.focus = foc }
changeFocus p x = guard (p x) >> return x
chFocus :: (a -> Bool) -> W.Stack a -> [W.Stack a]
@@ -105,15 +139,16 @@ getFocus :: Cursors b -> b
getFocus (Cons x) = getFocus $ W.focus x
getFocus (End x) = x
+-- This could be made more efficient, if the fact that the suffixes are grouped
focusTo :: (Eq t) => t -> Cursors t -> Maybe (Cursors t)
-focusTo x = listToMaybe . changeFocus ((x==) . getFocus)
+focusTo x = listToMaybe . filter ((x==) . getFocus) . changeFocus (const True)
--- | non-wrapping version of 'XMonad.StackSet.focusUp''
+-- | non-wrapping version of 'W.focusUp''
noWrapUp :: W.Stack t -> W.Stack t
noWrapUp (W.Stack t (l:ls) rs) = W.Stack l ls (t:rs)
noWrapUp x@(W.Stack _ [] _ ) = x
--- | non-wrapping version of 'XMonad.StackSet.focusDown''
+-- | non-wrapping version of 'W.focusDown''
noWrapDown :: W.Stack t -> W.Stack t
noWrapDown = reverseStack . noWrapUp . reverseStack
where reverseStack (W.Stack t ls rs) = W.Stack t rs ls
@@ -122,49 +157,63 @@ focusDepth :: Cursors t -> Int
focusDepth (Cons x) = 1 + focusDepth (W.focus x)
focusDepth (End _) = 0
-descend :: (W.Stack (Cursors a) -> W.Stack (Cursors a))-> Int-> Cursors a-> Cursors a
-descend f 1 (Cons x) = Cons $ f x
-descend f n (Cons x) | n > 1 = Cons $ descend f (pred n) `onFocus` x
-descend _ _ x = x
+descend :: Monad m =>(W.Stack (Cursors a) -> m (W.Stack (Cursors a)))-> Int-> Cursors a-> m (Cursors a)
+descend f 1 (Cons x) = Cons `liftM` f x
+descend f n (Cons x) | n > 1 = liftM Cons $ descend f (pred n) `onFocus` x
+descend _ _ x = return x
-onFocus :: (a -> a) -> W.Stack a -> W.Stack a
-onFocus f st = st { W.focus = f $ W.focus st }
+onFocus :: (Monad m) => (a1 -> m a1) -> W.Stack a1 -> m (W.Stack a1)
+onFocus f st = (\x -> st { W.focus = x}) `liftM` f (W.focus st)
+-- | @modifyLayer@ is used to change the focus at a given depth
modifyLayer :: (W.Stack (Cursors String) -> W.Stack (Cursors String)) -> Int -> X ()
-modifyLayer f depth = modifyCursors (return . descend f depth)
+modifyLayer f depth = modifyCursors (descend (return . f) depth)
+
+-- | @shiftModifyLayer@ is the same as 'modifyLayer', but also shifts the
+-- currently focused window to the new workspace
+shiftModifyLayer :: (W.Stack (Cursors String) -> W.Stack (Cursors WorkspaceId))-> Int-> X ()
+shiftModifyLayer f = modifyLayer' $ \st -> do
+ let st' = f st
+ windows $ W.shift $ getFocus (Cons st')
+ return st'
+
+-- | @shiftLayer@ is the same as 'shiftModifyLayer', but the focus remains on
+-- the current workspace.
+shiftLayer :: (W.Stack (Cursors String) -> W.Stack (Cursors WorkspaceId))-> Int-> X ()
+shiftLayer f = modifyLayer' $ \st -> do
+ windows $ W.shift $ getFocus $ Cons $ f st
+ return st
+
+-- | example usages are 'shiftLayer' and 'shiftModifyLayer'
+modifyLayer' :: (W.Stack (Cursors String) -> X (W.Stack (Cursors String))) -> Int -> X ()
+modifyLayer' f depth = modifyCursors (descend f depth)
modifyCursors :: (Cursors String -> X (Cursors String)) -> X ()
-modifyCursors = sendMessage . ChangeCursors
-
-currentWs :: X WorkspaceId
-currentWs = gets $ W.tag . W.workspace . W.current . windowset
+modifyCursors = sendMessage . ChangeCursors . (liftM2 (>>) updateXMD return <=<)
-data WorkspaceCursors a = WorkspaceCursors Bool (Cursors String) deriving (Typeable,Read,Show)
+data WorkspaceCursors a = WorkspaceCursors (Cursors String)
+ deriving (Typeable,Read,Show)
--- | WorkspaceCursors is implemented as a layout modifier, since that state is
--- serialized, and easily modified (with sendMessage)
-workspaceCursors :: Cursors String -> WorkspaceCursors a
-workspaceCursors = WorkspaceCursors False
+-- | The state is stored in the 'WorkspaceCursors' layout modifier. Put this as
+-- your outermost modifier, unless you want different cursors at different
+-- times (using "XMonad.Layout.MultiToggle")
+workspaceCursors :: Cursors String -> l a -> ModifiedLayout WorkspaceCursors l a
+workspaceCursors = ModifiedLayout . WorkspaceCursors
-data ChangeCursors = ChangeCursors {
- unWrap :: Cursors String -> X (Cursors String)
- } deriving (Typeable)
+data ChangeCursors = ChangeCursors { unWrap :: Cursors String -> X (Cursors String) }
+ deriving (Typeable)
instance Message ChangeCursors
+updateXMD :: Cursors WorkspaceId -> X ()
+updateXMD cs = do
+ changed <- gets $ (getFocus cs /=) . W.currentTag . windowset
+ when changed $ windows $ W.greedyView $ getFocus cs
+
instance LayoutModifier WorkspaceCursors a where
- redoLayout (WorkspaceCursors False cs) _ _ arrs = do
- cws <- currentWs
- return (arrs,do
- guard (getFocus cs /= cws)
- fmap (WorkspaceCursors True) $ focusTo cws cs)
-
- redoLayout (WorkspaceCursors _ cs) _ _ arrs = do
- cws <- currentWs
- -- redundant check to avoid switching workspaces
- unless (getFocus cs == cws) $ windows $ W.greedyView (getFocus cs)
- return (arrs,Nothing)
-
- handleMess (WorkspaceCursors prevMod cs) m =
- let wrap x = WorkspaceCursors (max prevMod (x /= cs)) x
- in sequenceA $ fmap wrap . ($ cs) . unWrap <$> fromMessage m
+ redoLayout (WorkspaceCursors cs) _ _ arrs = do
+ cws <- gets $ W.currentTag . windowset
+ return (arrs,WorkspaceCursors <$> focusTo cws cs)
+
+ handleMess (WorkspaceCursors cs) m =
+ sequenceA $ fmap WorkspaceCursors . ($ cs) . unWrap <$> fromMessage m