aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--XMonad/Layout/BoringWindows.hs121
1 files changed, 90 insertions, 31 deletions
diff --git a/XMonad/Layout/BoringWindows.hs b/XMonad/Layout/BoringWindows.hs
index 54d7b91..f56a004 100644
--- a/XMonad/Layout/BoringWindows.hs
+++ b/XMonad/Layout/BoringWindows.hs
@@ -1,5 +1,5 @@
-{-# OPTIONS_GHC -fglasgow-exts #-} -- For deriving Data/Typeable
-
+{-# LANGUAGE TypeSynonymInstances, MultiParamTypeClasses, DeriveDataTypeable #-}
+{-# LANGUAGE PatternGuards, FlexibleContexts, FlexibleInstances #-}
-----------------------------------------------------------------------------
-- |
-- Module : XMonad.Layout.BoringWindows
@@ -15,51 +15,110 @@
-----------------------------------------------------------------------------
module XMonad.Layout.BoringWindows (
- boringWindows,
+ -- * Usage
+ -- $usage
+ boringWindows, boringAuto,
markBoring, clearBoring,
- focusUp, focusDown
+ focusUp, focusDown,
+
+ UpdateBoring(UpdateBoring),
+ BoringMessage(Replace,Merge),
+ BoringWindows()
) where
-import XMonad hiding (Point)
+import XMonad.Layout.LayoutModifier(ModifiedLayout(..),
+ LayoutModifier(handleMessOrMaybeModifyIt, redoLayout))
+import XMonad(Typeable, LayoutClass, Message, X, fromMessage,
+ sendMessage, windows, withFocused, Window)
+import Control.Applicative((<$>))
+import Control.Monad(Monad(return, (>>)))
+import Data.List((\\), union)
+import Data.Maybe(Maybe(..), maybe, fromMaybe, listToMaybe,
+ maybeToList)
+import qualified Data.Map as M
import qualified XMonad.StackSet as W
-import XMonad.Layout.LayoutModifier
-import XMonad.Util.Invisible
+
+-- $usage
+-- You can use this module with the following in your
+-- @~\/.xmonad\/xmonad.hs@:
+--
+-- > import XMonad.Layout.BoringWindows
+--
+-- Then edit your @layoutHook@ by adding the layout modifier:
+--
+-- > myLayouts = boringWindows (Full ||| etc..)
+-- > main = xmonad defaultConfig { layoutHook = myLayouts }
+--
+-- Then to your keybindings, add:
+--
+-- > , ((modMask, xK_j), focusUp)
+-- > , ((modMask, xk_k), focusDown)
+--
+-- For more detailed instructions on editing the layoutHook see:
+--
+-- "XMonad.Doc.Extending#Editing_the_layout_hook"
+
data BoringMessage = FocusUp | FocusDown | IsBoring Window | ClearBoring
- deriving ( Read, Show, Typeable )
+ | Replace String [Window]
+ | Merge String [Window]
+ deriving ( Read, Show, Typeable )
+
instance Message BoringMessage
+-- | UpdateBoring is sent before attempting to view another boring window, so
+-- that layouts have a chance to mark boring windows.
+data UpdateBoring = UpdateBoring
+ deriving (Typeable)
+instance Message UpdateBoring
+
markBoring, clearBoring, focusUp, focusDown :: X ()
markBoring = withFocused (sendMessage . IsBoring)
clearBoring = sendMessage ClearBoring
-focusUp = sendMessage FocusUp
-focusDown = sendMessage FocusDown
+focusUp = sendMessage UpdateBoring >> sendMessage FocusUp
+focusDown = sendMessage UpdateBoring >> sendMessage FocusDown
-data BoringWindows a = BoringWindows (Invisible [] a) deriving ( Show, Read, Typeable )
+data BoringWindows a = BoringWindows
+ { namedBoring :: M.Map String [a] -- ^ store borings with a specific source
+ , chosenBoring :: [a] -- ^ user-chosen borings
+ , hiddenBoring :: Maybe [a] -- ^ maybe mark hidden windows
+ } deriving (Show,Read,Typeable)
boringWindows :: (LayoutClass l a, Eq a) => l a -> ModifiedLayout BoringWindows l a
-boringWindows = ModifiedLayout (BoringWindows (I []))
+boringWindows = ModifiedLayout (BoringWindows M.empty [] Nothing)
+
+-- | Mark windows that are not given rectangles as boring
+boringAuto :: (LayoutClass l a, Eq a) => l a -> ModifiedLayout BoringWindows l a
+boringAuto = ModifiedLayout (BoringWindows M.empty [] (Just []))
instance LayoutModifier BoringWindows Window where
- handleMessOrMaybeModifyIt (BoringWindows (I bs)) m
- | Just (IsBoring b) <- fromMessage m = return $ Just $ Left $ BoringWindows (I (b:bs))
- | Just ClearBoring <- fromMessage m = return $ Just $ Left $ BoringWindows (I [])
- | Just FocusUp <- fromMessage m = do windows $ W.modify' $ focusUp'
- return Nothing
+ redoLayout (b@BoringWindows { hiddenBoring = bs }) _r mst arrs = do
+ let bs' = W.integrate' mst \\ map fst arrs
+ return (arrs, Just $ b { hiddenBoring = const bs' <$> bs } )
+
+ handleMessOrMaybeModifyIt bst@(BoringWindows nbs cbs lbs) m
+ | Just (Replace k ws) <- fromMessage m
+ , maybe True (ws/=) (M.lookup k nbs) =
+ let nnb = if null ws then M.delete k nbs
+ else M.insert k ws nbs
+ in rjl bst { namedBoring = nnb }
+ | Just (Merge k ws) <- fromMessage m
+ , maybe True (not . null . (ws \\)) (M.lookup k nbs) =
+ rjl bst { namedBoring = M.insertWith union k ws nbs }
+ | Just (IsBoring w) <- fromMessage m , w `notElem` cbs =
+ rjl bst { chosenBoring = w:cbs }
+ | Just ClearBoring <- fromMessage m, not (null cbs) =
+ rjl bst { namedBoring = M.empty, chosenBoring = []}
+ | Just FocusUp <- fromMessage m =
+ do windows $ W.modify' $ skipBoring W.focusUp'
+ return Nothing
| Just FocusDown <- fromMessage m =
- do windows $ W.modify' (reverseStack . focusUp' . reverseStack)
+ do windows $ W.modify' $ skipBoring W.focusDown'
return Nothing
- where focusUp' (W.Stack t ls rs)
- | (a,l:ls') <- skipBoring ls = W.Stack l ls' (a++t:rs)
- | otherwise = case skipBoring (reverse (t:rs)++ls) of
- (a,x:xs) -> W.Stack x xs a
- _ -> W.Stack t ls rs
- skipBoring [] = ([],[])
- skipBoring (x:xs) | x `elem` bs = case skipBoring xs of
- (a,b) -> (x:a,b)
- | otherwise = ([],x:xs)
+ where skipBoring f st = fromMaybe st $ listToMaybe
+ $ filter ((`notElem` W.focus st:bs) . W.focus)
+ $ take (length $ W.integrate st)
+ $ iterate f st
+ bs = concat $ cbs:maybeToList lbs ++ M.elems nbs
+ rjl = return . Just . Left
handleMessOrMaybeModifyIt _ _ = return Nothing
-
--- | reverse a stack: up becomes down and down becomes up.
-reverseStack :: W.Stack a -> W.Stack a
-reverseStack (W.Stack t ls rs) = W.Stack t rs ls