aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--XMonad/Layout/Groups.hs83
1 files changed, 46 insertions, 37 deletions
diff --git a/XMonad/Layout/Groups.hs b/XMonad/Layout/Groups.hs
index 75658d7..fad1154 100644
--- a/XMonad/Layout/Groups.hs
+++ b/XMonad/Layout/Groups.hs
@@ -60,7 +60,7 @@ import qualified XMonad.StackSet as W
import XMonad.Util.Stack
-import Data.Maybe (isJust, isNothing, fromMaybe, catMaybes)
+import Data.Maybe (isJust, isNothing, fromMaybe, catMaybes, fromJust)
import Data.List ((\\))
import Control.Arrow ((>>>))
import Control.Applicative ((<$>))
@@ -99,7 +99,8 @@ import Control.Monad (forM)
-- need to manage 'Window's. This is obvious, when you think
-- about it.
group :: l Window -> l2 (Group l Window) -> Groups l l2 Window
-group l l2 = Groups l l2 emptyZ (U 0 0)
+group l l2 = Groups l l2 startingGroups (U 1 0)
+ where startingGroups = fromJust $ singletonZ $ G (ID (U 0 0) l) emptyZ
-- * Stuff with unique keys
@@ -167,7 +168,7 @@ data Groups l l2 a = Groups { -- | The starting layout for new groups
-- | The layout for placing each group on the screen
, partitioner :: l2 (Group l a)
-- | The window groups
- , groups :: Zipper (Group l a)
+ , groups :: W.Stack (Group l a)
-- | A seed for generating unique ids
, seed :: Uniq
}
@@ -203,7 +204,10 @@ instance Message GroupsMessage
modifyGroups :: (Zipper (Group l a) -> Zipper (Group l a))
-> Groups l l2 a -> Groups l l2 a
-modifyGroups f g = g { groups = f $ groups g }
+modifyGroups f g = let (seed', id:_) = gen (seed g)
+ defaultGroups = fromJust $ singletonZ $ G (ID id $ baseLayout g) emptyZ
+ in g { groups = fromMaybe defaultGroups . f . Just $ groups g
+ , seed = seed' }
-- ** Readaptation
@@ -213,21 +217,23 @@ modifyGroups f g = g { groups = f $ groups g }
-- ignores changes in window ordering, and tries to react to any
-- other stack changes as gracefully as possible.
readapt :: Eq a => Zipper a -> Groups l l2 a -> Groups l l2 a
-readapt Nothing g = g { groups = Nothing }
-readapt (Just s) g = let f = W.focus s
- (seed', id:_) = gen $ seed g
- g' = g { seed = seed' }
- in flip modifyGroups g' $ mapZ_ (onZipper $ removeDeleted s)
- >>> filterZ_ (isJust . gZipper)
- >>> findNewWindows (W.integrate s)
- >>> addWindows (ID id $ baseLayout g)
- >>> focusGroup f
- >>> onFocusedZ (onZipper $ focusWindow f)
+readapt z g = let mf = getFocusZ z
+ (seed', id:_) = gen $ seed g
+ g' = g { seed = seed' }
+ in flip modifyGroups g' $ mapZ_ (onZipper $ removeDeleted z)
+ >>> filterKeepLast (isJust . gZipper)
+ >>> findNewWindows (W.integrate' z)
+ >>> addWindows (ID id $ baseLayout g)
+ >>> focusGroup mf
+ >>> onFocusedZ (onZipper $ focusWindow mf)
+ where filterKeepLast _ Nothing = Nothing
+ filterKeepLast f z@(Just s) = maybe (singletonZ $ W.focus s) Just
+ $ filterZ_ f z
-- | Remove the windows from a group which are no longer present in
-- the stack.
-removeDeleted :: Eq a => W.Stack a -> Zipper a -> Zipper a
-removeDeleted s = filterZ_ (flip elemZ $ Just s)
+removeDeleted :: Eq a => Zipper a -> Zipper a -> Zipper a
+removeDeleted z = filterZ_ (flip elemZ z)
-- | Identify the windows not already in a group.
findNewWindows :: Eq a => [a] -> Zipper (Group l a)
@@ -243,12 +249,14 @@ addWindows _ (z, as) = onFocusedZ (onZipper add) z
where add z = foldl (flip insertUpZ) z as
-- | Focus the group containing the given window
-focusGroup :: Eq a => a -> Zipper (Group l a) -> Zipper (Group l a)
-focusGroup a = fromTags . map (tagBy $ elemZ a . gZipper) . W.integrate'
+focusGroup :: Eq a => Maybe a -> Zipper (Group l a) -> Zipper (Group l a)
+focusGroup Nothing = id
+focusGroup (Just a) = fromTags . map (tagBy $ elemZ a . gZipper) . W.integrate'
-- | Focus the given window
-focusWindow :: Eq a => a -> Zipper a -> Zipper a
-focusWindow a = fromTags . map (tagBy (==a)) . W.integrate'
+focusWindow :: Eq a => Maybe a -> Zipper a -> Zipper a
+focusWindow Nothing = id
+focusWindow (Just a) = fromTags . map (tagBy (==a)) . W.integrate'
-- * Interface
@@ -258,18 +266,18 @@ focusWindow a = fromTags . map (tagBy (==a)) . W.integrate'
instance (LayoutClass l Window, LayoutClass l2 (Group l Window))
=> LayoutClass (Groups l l2) Window where
- description (Groups b p gs _) = s1++" by "++s2
- where s1 = fromMaybe (description b) $ fmap (description . gLayout) $ getFocusZ gs
+ description (Groups _ p gs _) = s1++" by "++s2
+ where s1 = description $ gLayout $ W.focus gs
s2 = description p
runLayout ws@(W.Workspace _ _l z) r = let l = readapt z _l in
do (areas, mpart') <- runLayout ws { W.layout = partitioner l
- , W.stack = groups l } r
+ , W.stack = Just $ groups l } r
results <- forM areas $ \(g, r') -> runLayout ws { W.layout = gLayout g
, W.stack = gZipper g } r'
- let hidden = map gLayout (W.integrate' $ groups l) \\ map (gLayout . fst) areas
+ let hidden = map gLayout (W.integrate $ groups l) \\ map (gLayout . fst) areas
hidden' <- mapM (flip handleMessage $ SomeMessage Hide) hidden
let placements = concatMap fst results
@@ -283,7 +291,7 @@ instance (LayoutClass l Window, LayoutClass l2 (Group l Window))
handleMessage l@(Groups _ p gs _) sm | Just (ToAll sm') <- fromMessage sm
= do mp' <- handleMessage p sm'
- mg's <- mapZM_ (handle sm') gs
+ mg's <- mapZM_ (handle sm') $ Just gs
return $ maybeMakeNew l mp' $ W.integrate' mg's
where handle sm (G l _) = handleMessage l sm
@@ -292,7 +300,7 @@ instance (LayoutClass l Window, LayoutClass l2 (Group l Window))
-- of doing this?
in handleMessage l $ SomeMessage $ ToAll sm
- handleMessage l@(Groups _ _ z@(Just _) _) sm = case fromMessage sm of
+ handleMessage l@(Groups _ _ z _) sm = case fromMessage sm of
Just (ToFocused sm') -> do mg's <- W.integrate' <$> handleOnFocused sm' z
return $ maybeMakeNew l Nothing mg's
Just (ToGroup i sm') -> do mg's <- handleOnIndex i sm' z
@@ -303,34 +311,32 @@ instance (LayoutClass l Window, LayoutClass l2 (Group l Window))
Just Refocus -> refocus l >> return (Just l)
Just _ -> return Nothing
Nothing -> handleMessage l $ SomeMessage (ToFocused sm)
- where handleOnFocused sm z = mapZM step z
+ where handleOnFocused sm z = mapZM step $ Just z
where step True (G l _) = handleMessage l sm
step False _ = return Nothing
- handleOnIndex i sm z = mapM step $ zip [0..] $ W.integrate' z
+ handleOnIndex i sm z = mapM step $ zip [0..] $ W.integrate z
where step (j, (G l _)) | i == j = handleMessage l sm
step _ = return Nothing
- handleMessage _ _ = return Nothing
-
justMakeNew :: Groups l l2 a -> Maybe (l2 (Group l a)) -> [Maybe (WithID l a)]
-> Maybe (Groups l l2 a)
justMakeNew g mpart' ml's = Just g { partitioner = fromMaybe (partitioner g) mpart'
, groups = combine (groups g) ml's }
where combine z ml's = let table = map (\(ID id a) -> (id, a)) $ catMaybes ml's
- in flip mapZ_ z $ \(G (ID id l) ws) -> case lookup id table of
+ in flip mapS_ z $ \(G (ID id l) ws) -> case lookup id table of
Nothing -> G (ID id l) ws
Just l' -> G (ID id l') ws
+ mapS_ f = fromJust . mapZ_ f . Just
maybeMakeNew :: Groups l l2 a -> Maybe (l2 (Group l a)) -> [Maybe (WithID l a)]
-> Maybe (Groups l l2 a)
-maybeMakeNew _ Nothing [] = Nothing
maybeMakeNew _ Nothing ml's | all isNothing ml's = Nothing
maybeMakeNew g mpart' ml's = justMakeNew g mpart' ml's
refocus :: Groups l l2 Window -> X ()
-refocus g = case getFocusZ (groups g) >>= (getFocusZ . gZipper)
+refocus g = case getFocusZ $ gZipper $ W.focus $ groups g
of Just w -> focus w
Nothing -> return ()
@@ -356,11 +362,14 @@ type ModifySpec = forall l. WithID l Window
-- | Apply a ModifySpec.
applySpec :: ModifySpec -> Groups l l2 Window -> Maybe (Groups l l2 Window)
applySpec f g = let (seed', id:ids) = gen $ seed g
- gs' = f (ID id $ baseLayout g) (groups g)
- gs'' = fromTags $ snd $ foldr reID ((ids, []), []) $ toTags gs'
- in case groups g == gs' of
+ g' = flip modifyGroups g $ f (ID id $ baseLayout g)
+ >>> toTags
+ >>> foldr reID ((ids, []), [])
+ >>> snd
+ >>> fromTags
+ in case groups g == groups g' of
True -> Nothing
- False -> Just g { groups = gs'', seed = seed' }
+ False -> Just g' { seed = seed' }
where reID eg ((id:ids, seen), egs)
= let myID = getID $ gLayout $ fromE eg