diff options
-rw-r--r-- | XMonad/Layout/LimitWindows.hs | 68 | ||||
-rw-r--r-- | XMonad/Layout/Selective.hs | 118 | ||||
-rw-r--r-- | tests/test_Selective.hs | 8 | ||||
-rw-r--r-- | xmonad-contrib.cabal | 1 |
4 files changed, 69 insertions, 126 deletions
diff --git a/XMonad/Layout/LimitWindows.hs b/XMonad/Layout/LimitWindows.hs index 61b6164..c3d3def 100644 --- a/XMonad/Layout/LimitWindows.hs +++ b/XMonad/Layout/LimitWindows.hs @@ -1,8 +1,9 @@ -{-# LANGUAGE FlexibleInstances, MultiParamTypeClasses, DeriveDataTypeable #-} +{-# LANGUAGE FlexibleInstances, MultiParamTypeClasses, DeriveDataTypeable, NamedFieldPuns, PatternGuards #-} ----------------------------------------------------------------------------- -- | -- Module : XMonad.Layout.LimitWindows -- Copyright : (c) 2009 Adam Vogt +-- (c) 2009 Max Rabkin -- wrote limitSelect -- License : BSD-style (see xmonad/LICENSE) -- -- Maintainer : vogt.adam@gmail.com @@ -17,17 +18,19 @@ module XMonad.Layout.LimitWindows ( -- * Usage -- $usage - -- Layout Modifiers - limitWindows,limitSlice, + -- * Layout Modifiers + limitWindows,limitSlice,limitSelect, - -- Change the number of windows + -- * Change the number of windows increaseLimit,decreaseLimit,setLimit ) where import XMonad.Layout.LayoutModifier import XMonad import qualified XMonad.StackSet as W +import XMonad.Layout (IncMasterN (..)) import Control.Monad((<=<),guard) +import Control.Applicative((<$>)) import Data.Maybe(fromJust) -- $usage @@ -66,6 +69,12 @@ limitWindows n = ModifiedLayout (LimitWindows FirstN n) limitSlice :: Int -> l a -> ModifiedLayout LimitWindows l a limitSlice n = ModifiedLayout (LimitWindows Slice n) +-- | Only display the first @m@ windows and @r@ others. +-- The @IncMasterN@ message will change @m@, as well as passing it onto the +-- underlying layout. +limitSelect :: Int -> Int -> l a -> ModifiedLayout Selection l a +limitSelect m r = ModifiedLayout Sel{ nMaster=m, start=m, nRest=r } + data LimitWindows a = LimitWindows SliceStyle Int deriving (Read,Show) data SliceStyle = FirstN | Slice deriving (Read,Show) @@ -81,7 +90,7 @@ instance LayoutModifier LimitWindows a where app f x = guard (f x /= x) >> return (f x) modifyLayout (LimitWindows style n) ws r = - runLayout ws { W.stack = f n `fmap` W.stack ws } r + runLayout ws { W.stack = f n <$> W.stack ws } r where f = case style of FirstN -> firstN Slice -> slice @@ -99,3 +108,52 @@ slice n (W.Stack f u d) = unusedU = max 0 $ nu - length u nd = div (n - 1) 2 nu = uncurry (+) $ divMod (n - 1) 2 + +data Selection a = Sel { nMaster :: Int, start :: Int, nRest :: Int } + deriving (Read, Show, Eq) + +instance LayoutModifier Selection a where + modifyLayout s w r = + runLayout (w { W.stack = updateAndSelect s <$> W.stack w }) r + + pureModifier sel _ stk wins = (wins, update sel <$> stk) + + pureMess sel m + | Just f <- unLC <$> fromMessage m = + Just $ sel { nRest = max 0 (f (nMaster sel + nRest sel) - nMaster sel) } + | Just (IncMasterN n) <- fromMessage m = + Just $ sel { nMaster = max 0 (nMaster sel + n) } + | otherwise = + Nothing + +select :: Selection l -> W.Stack a -> W.Stack a +select (Sel { nMaster, start, nRest }) stk + | lups < nMaster + = stk { W.down=take (nMaster - lups - 1) downs ++ + (take nRest . drop (start - lups - 1) $ downs) } + | otherwise + = stk { W.up=reverse (take nMaster ups ++ drop start ups), + W.down=take (nRest - (lups - start) - 1) downs } + where + downs = W.down stk + ups = reverse $ W.up stk + lups = length ups + +updateStart :: Selection l -> W.Stack a -> Int +updateStart (Sel { nMaster, start, nRest }) stk + | lups < nMaster -- the focussed window is in the master pane + = start `min` (lups + ldown - nRest + 1) `max` nMaster + | otherwise + = start `min` lups + `max` (lups - nRest + 1) + `min` (lups + ldown - nRest + 1) + `max` nMaster + where + lups = length $ W.up stk + ldown = length $ W.down stk + +update :: Selection l -> W.Stack a -> Selection a +update sel stk = sel { start=updateStart sel stk } + +updateAndSelect :: Selection l -> W.Stack a -> W.Stack a +updateAndSelect sel stk = select (update sel stk) stk diff --git a/XMonad/Layout/Selective.hs b/XMonad/Layout/Selective.hs deleted file mode 100644 index 2999d34..0000000 --- a/XMonad/Layout/Selective.hs +++ /dev/null @@ -1,118 +0,0 @@ ------------------------------------------------------------------------------ --- | --- Module : XMonad.Layout.Selective --- Copyright : (c) 2009 Max Rabkin --- License : BSD3-style (see LICENSE) --- --- Maintainer : Max Rabkin <max.rabkin@gmail.com> --- Stability : unstable --- Portability : unportable --- --- Provides a layout modifier that only shows the master pane and windows --- around the focussed window. --- ------------------------------------------------------------------------------ - -{-# LANGUAGE MultiParamTypeClasses, - FlexibleInstances, - NoMonomorphismRestriction, - NamedFieldPuns #-} - -module XMonad.Layout.Selective ( - -- * Description - -- $description - -- * Usage - -- $usage - - -- The Layout Modifier - selective - ) where - -import XMonad.Core -import XMonad.StackSet -import XMonad.Layout (IncMasterN (..)) -import XMonad.Layout.LayoutModifier -import Control.Applicative ((<$>)) - --- $description --- Selective is a layout modifier which limits the number of windows on screen. --- The first @n@ windows ("the master pane", which may correspond to the --- master pane of the underlying layout) plus several others are shown, such --- that the focussed window is always visible. Windows are not moved until a --- hidden window gains focus. - --- $usage --- To use this module, add the following import to @~\/.xmonad\/xmonad.hs@: --- --- > import XMonad.Layout.Selective --- --- > myLayout = (selective 1 3 $ Tall 1 0.03 0.5) ||| Full ||| RandomOtherLayout... --- > main = xmonad defaultConfig { layoutHook = myLayout } --- --- The layout modifier accepts the IncMasterN message to change the number of --- windows in the "master pane". --- --- For detailed instructions on editing your key bindings, see --- "XMonad.Doc.Extending#Editing_key_bindings". --- --- See also 'XMonad.Layout.BoringWindows.boringAuto' for keybindings that skip --- the hidden windows. - --- invariant: 0 <= nMaster <= start; 1 <= nRest -data Selection = Sel { nMaster :: Int, start :: Int, nRest :: Int } - deriving (Read, Show, Eq) - -select :: Selection -> Stack a -> Stack a -select (Sel { nMaster, start, nRest }) stk - | lups < nMaster - = stk { down=take (nMaster - lups - 1) downs ++ - (take nRest . drop (start - lups - 1) $ downs) } - | otherwise - = stk { up=reverse (take nMaster ups ++ drop start ups), - down=take (nRest - (lups - start) - 1) downs } - where - downs = down stk - ups = reverse $ up stk - lups = length ups - -updateStart :: Selection -> Stack a -> Int -updateStart (Sel { nMaster, start, nRest }) stk - | lups < nMaster -- the focussed window is in the master pane - = start `min` (lups + ldown - nRest + 1) `max` nMaster - | otherwise - = start `min` lups - `max` (lups - nRest + 1) - `min` (lups + ldown - nRest + 1) - `max` nMaster - where - lups = length $ up stk - ldown = length $ down stk - -update :: Selection -> Stack a -> Selection -update sel stk = sel { start=updateStart sel stk } - -updateAndSelect :: Selection -> Stack a -> Stack a -updateAndSelect sel stk = select (update sel stk) stk - -data Selective a = Selective Selection - deriving (Read, Show) - -instance LayoutModifier Selective a where - modifyLayout (Selective s) w r = - runLayout (w { stack = updateAndSelect s <$> stack w }) r - - pureModifier (Selective sel) _ stk wins = (wins, Selective . update sel <$> stk) - - pureMess (Selective s) m = Selective . incmastern <$> fromMessage m - where - incmastern (IncMasterN n) = - let nm = (nMaster s + n) `max` 0 - in if nMaster s == start s - then s { nMaster = nm, start = nm } - else s { nMaster = nm } - --- | Only display the first @m@ windows and @r@ others. --- The @IncMasterN@ message will change @m@, as well as passing it onto the --- underlying layout. -selective :: Int -> Int -> l a -> ModifiedLayout Selective l a -selective m r = ModifiedLayout . Selective $ Sel { nMaster=m, start=m, nRest=r } diff --git a/tests/test_Selective.hs b/tests/test_Selective.hs index 4de4632..15ea5c0 100644 --- a/tests/test_Selective.hs +++ b/tests/test_Selective.hs @@ -1,6 +1,10 @@ {-# LANGUAGE ScopedTypeVariables, FlexibleInstances #-} -import XMonad.Layout.Selective +-- Tests for limitSelect-related code in L.LimitWindows. +-- To run these tests, export (select,update,Selection(..),updateAndSelect) from +-- L.LimitWindows. + +import XMonad.Layout.LimitWindows import XMonad.StackSet hiding (focusUp, focusDown) import Control.Applicative ((<$>)) import Test.QuickCheck @@ -13,7 +17,7 @@ instance Arbitrary (Stack Int) where return $ Stack { up=[xs-1,xs-2..0], focus=xs, down=[xs+1..xs+ys] } coarbitrary = undefined -instance Arbitrary Selection where +instance Arbitrary (Selection a) where arbitrary = do nm <- arbNat st <- arbNat diff --git a/xmonad-contrib.cabal b/xmonad-contrib.cabal index e425de0..e589eeb 100644 --- a/xmonad-contrib.cabal +++ b/xmonad-contrib.cabal @@ -183,7 +183,6 @@ library XMonad.Layout.ResizableTile XMonad.Layout.ResizeScreen XMonad.Layout.Roledex - XMonad.Layout.Selective XMonad.Layout.Simplest XMonad.Layout.SimpleDecoration XMonad.Layout.SimpleFloat |