aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--XMonad/Layout/Mosaic.hs219
1 files changed, 112 insertions, 107 deletions
diff --git a/XMonad/Layout/Mosaic.hs b/XMonad/Layout/Mosaic.hs
index ee38dc1..ec72611 100644
--- a/XMonad/Layout/Mosaic.hs
+++ b/XMonad/Layout/Mosaic.hs
@@ -1,4 +1,4 @@
-{-# LANGUAGE FlexibleInstances, MultiParamTypeClasses, DeriveDataTypeable #-}
+{-# LANGUAGE FlexibleInstances, MultiParamTypeClasses, DeriveDataTypeable, PatternGuards #-}
-----------------------------------------------------------------------------
-- |
-- Module : XMonad.Layout.Mosaic
@@ -17,24 +17,27 @@
module XMonad.Layout.Mosaic (
-- * Usage
-- $usage
- Mosaic(Mosaic)
- ,Aspect(..)
- ,shallower
- ,steeper
- ,growMaster
- ,shrinkMaster
+ Aspect(..)
+ ,mosaic
,changeMaster
+ ,changeFocused
)
where
import Prelude hiding (sum)
import XMonad(Typeable,
- LayoutClass(doLayout , pureMessage, description), Message,
- fromMessage, splitHorizontallyBy, splitVerticallyBy, Rectangle)
-import XMonad.StackSet(integrate)
+ LayoutClass(doLayout, handleMessage, pureMessage, description),
+ Message, X, fromMessage, withWindowSet, Resize(..),
+ splitHorizontallyBy, splitVerticallyBy, sendMessage, Rectangle)
+import qualified XMonad.StackSet as W
+import Control.Arrow(Control.Arrow.Arrow(second, first))
+import Control.Monad(mplus)
import Data.Foldable(Foldable(foldMap), sum)
-import Data.Monoid(Monoid(mappend, mempty))
+import Data.Function(on)
+import Data.List(sortBy)
+import Data.Monoid(Monoid(mempty, mappend))
+
-- $usage
-- You can use this module with the following in your @~\/.xmonad\/xmonad.hs@:
@@ -43,18 +46,18 @@ import Data.Monoid(Monoid(mappend, mempty))
--
-- Then edit your @layoutHook@ by adding the Mosaic layout:
--
--- > myLayouts = Mosaic (take 5 $ iterate (*0.7) 1) ||| Mosaic [3,1,1,1,1,1] ||| Full ||| etc..
--- > main = xmonad defaultConfig { layoutHook = myLayouts }
+-- > myLayouts = mosaic 2 [3,2] ||| Full ||| etc..
+-- > main = xmonad $ defaultConfig { layoutHook = myLayouts }
--
--- Unfortunately, infinite lists break serialization, so don't use them.
+-- Unfortunately, infinite lists break serialization, so don't use them. And if
+-- the list is too short, it is extended with @++ repeat 1@, which covers the
+-- main use case.
--
-- To change the choice in aspect ratio and the relative sizes of windows, add
-- to your keybindings:
--
-- > , ((modMask, xK_a), sendMessage Taller)
-- > , ((modMask, xK_z), sendMessage Wider)
--- > , ((modMask, xK_h), sendMessage Shrink >> sendMessage (SlopeMod shallower))
--- > , ((modMask, xK_l), sendMessage Expand >> sendMessage (SlopeMod steeper))
--
-- > , ((modMask, xK_r), sendMessage Reset)
--
@@ -71,90 +74,88 @@ data Aspect
instance Message Aspect
-data Mosaic a
- {- | The relative magnitudes (the sign is ignored) of the rational numbers
- - provided determine the relative areas that the windows receive. The
- - first number represents the size of the master window, the second is for
- - the next window in the stack, and so on. Windows without a list element
- - are hidden.
- -}
- = Mosaic [Rational]
- -- override the aspect? current index, maximum index
- | MosaicSt Bool Rational Int [Rational]
- deriving (Read, Show)
+-- | The relative magnitudes (the sign is ignored) of the rational numbers in
+-- the second argument determine the relative areas that the windows receive.
+-- The first number represents the size of the master window, the second is for
+-- the next window in the stack, and so on.
+--
+-- The list is extended with @++ repeat 1@, so @mosaic 1.5 []@ is like a
+-- resizable grid.
+--
+-- The first parameter is the multiplicative factor to use when responding to
+-- the 'Expand' message.
+mosaic :: Rational -> [Rational] -> Mosaic a
+mosaic = Mosaic Nothing
+
+data Mosaic a = -- | True to override the aspect, current index, maximum index
+ Mosaic (Maybe(Bool,Rational,Int)) Rational [Rational] deriving (Read,Show)
instance LayoutClass Mosaic a where
description = const "Mosaic"
- pureMessage (Mosaic _ss) _ms = Nothing
- pureMessage (MosaicSt _ ix mix ss) ms = fromMessage ms >>= ixMod
- where ixMod Taller | rix >= mix = Nothing
- | otherwise = Just $ MosaicSt False (succ ix) mix ss
- ixMod Wider | rix <= 0 = Nothing
- | otherwise = Just $ MosaicSt False (pred ix) mix ss
- ixMod Reset = Just $ Mosaic ss
- ixMod (SlopeMod f) = Just $ MosaicSt False ix mix (f ss)
- rix = round ix
-
- doLayout (Mosaic ss) r st = return (zip (integrate st) rect, newLayout)
- where rects = splits (length $ integrate st) r ss
- lrects = length rects
- rect = rects !! (lrects `div` 2)
- newLayout = Just $ MosaicSt True (fromIntegral lrects / 2) (pred lrects) ss
-
- doLayout (MosaicSt override ix mix ss) r st
- = return (zip (integrate st) rect, newLayout)
- where rects = splits (length $ integrate st) r ss
- lrects = length rects
- nix = if mix == 0 || override then fromIntegral $ lrects `div` 2
- else max 0 $ min (fromIntegral $ pred lrects)
- $ fromIntegral (pred lrects) * ix / fromIntegral mix
- rect = rects !! round nix
- newLayout = Just $ MosaicSt override nix (pred lrects) ss
+ pureMessage (Mosaic Nothing _ _) _ = Nothing
+ pureMessage (Mosaic (Just(_,ix,mix)) delta ss) ms = fromMessage ms >>= ixMod
+ where ixMod Taller | round ix >= mix = Nothing
+ | otherwise = Just $ Mosaic (Just(False,succ ix,mix)) delta ss
+ ixMod Wider | round ix <= (0::Integer) = Nothing
+ | otherwise = Just $ Mosaic (Just(False,pred ix,mix)) delta ss
+ ixMod Reset = Just $ Mosaic Nothing delta ss
+ ixMod (SlopeMod f) = Just $ Mosaic (Just(False,ix,mix)) delta (f ss)
+
+ handleMessage l@(Mosaic _ delta _) ms
+ | Just Expand <- fromMessage ms = changeFocused (*delta) >> return Nothing
+ | Just Shrink <- fromMessage ms = changeFocused (/delta) >> return Nothing
+ | otherwise = return $ pureMessage l ms
+
+ doLayout (Mosaic state delta ss) r st = let
+ ssExt = zipWith const (ss ++ repeat 1) $ W.integrate st
+ rects = splits r ssExt
+ nls = length rects
+ fi = fromIntegral
+ nextIx (ov,ix,mix)
+ | mix <= 0 || ov = fromIntegral $ nls `div` 2
+ | otherwise = max 0 $ (*fi (pred nls)) $ min 1 $ ix / fi mix
+ rect = rects !! maybe (nls `div` 2) round (nextIx `fmap` state)
+ state' = fmap (\x@(ov,_,_) -> (ov,nextIx x,pred nls)) state
+ `mplus` Just (True,fromIntegral nls / 2,pred nls)
+ ss' | and $ zipWith (==) ss ssExt = ss
+ | otherwise = ssExt
+ in return (zip (W.integrate st) rect, Just $ Mosaic state' delta ss')
-- | These sample functions are meant to be applied to the list of window sizes
-- through the 'SlopeMod' message.
+changeMaster :: (Rational -> Rational) -> X ()
+changeMaster = sendMessage . SlopeMod . onHead
+
+-- | Apply a function to the Rational that represents the currently focused
+-- window.
--
--- Steeper and shallower scale the ratios of successive windows.
---
--- growMaster and shrinkMaster just increase and decrease the size of the first
--- element, and thus they change the layout very similarily to the standard
--- 'Expand' or 'Shrink' for the 'Tall' layout.
+-- 'Expand' and 'Shrink' messages are responded to with @changeFocused
+-- (*delta)@ or @changeFocused (delta/)@ where @delta@ is the first argument to
+-- 'mosaic'.
--
--- It may be possible to resize the specific focused window; however the same
--- result could probably be achieved by promoting it, or moving it to a higher
--- place in the list of windows; when you have a decreasing list of window
--- sizes, the change in position will also result in a change in size.
-
-steeper :: [Rational] -> [Rational]
-steeper [] = []
-steeper xs = map (subtract (minimum xs*0.8)) xs
-
-shallower :: [Rational] -> [Rational]
-shallower [] = []
-shallower xs = map (+(minimum xs*2)) xs
-
-growMaster :: [Rational] -> [Rational]
-growMaster = changeMaster 2
-
-shrinkMaster :: [Rational] -> [Rational]
-shrinkMaster = changeMaster 0.5
-
--- | Multiply the area of the current master by a specified ratio
-changeMaster :: Rational -> [Rational] -> [Rational]
-changeMaster _ [] = []
-changeMaster f (x:xs) = f*x:xs
-
-splits :: Int -> Rectangle -> [Rational] -> [[Rectangle]]
-splits num rect = splitsL rect . makeTree . normalize
- . map abs . reverse . take num
-
--- recursively enumerate splits
-splitsL :: Rectangle -> Tree Rational -> [[Rectangle]]
+-- This is exported because other functions (ex. @const 1@, @(+1)@) may be
+-- useful to apply to the current area.
+changeFocused :: (Rational -> Rational) -> X ()
+changeFocused f = withWindowSet $ sendMessage . SlopeMod
+ . maybe id (mulIx . length . W.up)
+ . W.stack . W.workspace . W.current
+ where mulIx i = uncurry (++) . second (onHead f) . splitAt i
+
+onHead :: (a -> a) -> [a] -> [a]
+onHead f = uncurry (++) . first (fmap f) . splitAt 1
+
+splits :: Rectangle -> [Rational] -> [[Rectangle]]
+splits rect = map (reverse . map snd . sortBy (compare `on` fst))
+ . splitsL rect . makeTree snd . zip [1..]
+ . normalize . reverse . map abs
+
+splitsL :: Rectangle -> Tree (Int,Rational) -> [[(Int,Rectangle)]]
splitsL _rect Empty = []
-splitsL rect (Leaf _) = [[rect]]
+splitsL rect (Leaf (x,_)) = [[(x,rect)]]
splitsL rect (Branch l r) = do
- let mkSplit f = f (sum l / (sum l + sum r)) rect
+ let mkSplit f = f ((sumSnd l /) $ sumSnd l + sumSnd r) rect
+ sumSnd = sum . fmap snd
(rl,rr) <- map mkSplit [splitVerticallyBy,splitHorizontallyBy]
splitsL rl l `interleave` splitsL rr r
@@ -163,38 +164,42 @@ splitsL rect (Branch l r) = do
interleave :: [[a]] -> [[a]] -> [[a]]
interleave xs ys | lx > ly = zc xs (extend lx ys)
| otherwise = zc (extend ly xs) ys
- where lx = length xs
- ly = length ys
- zc = zipWith (++)
+ where lx = length xs
+ ly = length ys
+ zc = zipWith (++)
-extend :: Int -> [a] -> [a]
-extend n pat = do
- (p,e) <- zip pat $ replicate m True ++ repeat False
- [p | e] ++ replicate d p
- where (d,m) = n `divMod` length pat
+ extend :: Int -> [a] -> [a]
+ extend n pat = do
+ (p,e) <- zip pat $ replicate m True ++ repeat False
+ [p | e] ++ replicate d p
+ where (d,m) = n `divMod` length pat
normalize :: Fractional a => [a] -> [a]
-normalize x = let s = sum x
- in map (/s) x
+normalize x = let s = sum x in map (/s) x
data Tree a = Branch (Tree a) (Tree a) | Leaf a | Empty
- deriving (Show)
instance Foldable Tree where
foldMap _f Empty = mempty
foldMap f (Leaf x) = f x
foldMap f (Branch l r) = foldMap f l `mappend` foldMap f r
+instance Functor Tree where
+ fmap f (Leaf x) = Leaf $ f x
+ fmap f (Branch l r) = Branch (fmap f l) (fmap f r)
+ fmap _ Empty = Empty
+
instance Monoid (Tree a) where
mempty = Empty
mappend Empty x = x
mappend x Empty = x
mappend x y = Branch x y
-makeTree :: [Rational] -> Tree Rational
-makeTree [] = Empty
-makeTree [x] = Leaf x
-makeTree xs = Branch (makeTree a) (makeTree b)
- where ((a,b),_) = foldr w (([],[]),(0,0)) xs
- w n ((ls,rs),(l,r)) = if l > r then ((ls,n:rs),(l,n+r))
- else ((n:ls,rs),(n+l,r))
+makeTree :: (Num a1, Ord a1) => (a -> a1) -> [a] -> Tree a
+makeTree _ [] = Empty
+makeTree _ [x] = Leaf x
+makeTree f xs = Branch (makeTree f a) (makeTree f b)
+ where ((a,b),_) = foldr go (([],[]),(0,0)) xs
+ go n ((ls,rs),(l,r))
+ | l > r = ((ls,n:rs),(l,f n+r))
+ | otherwise = ((n:ls,rs),(f n+l,r))