aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--XMonad/Layout/GridVariants.hs166
-rw-r--r--xmonad-contrib.cabal1
2 files changed, 167 insertions, 0 deletions
diff --git a/XMonad/Layout/GridVariants.hs b/XMonad/Layout/GridVariants.hs
new file mode 100644
index 0000000..20c970d
--- /dev/null
+++ b/XMonad/Layout/GridVariants.hs
@@ -0,0 +1,166 @@
+{-# LANGUAGE DeriveDataTypeable, FlexibleInstances, MultiParamTypeClasses #-}
+
+----------------------------------------------------------------------
+-- |
+-- Module : XMonad.Layout.GridVariants
+-- Copyright : (c) Norbert Zeh
+-- License : BSD-style (see LICENSE)
+--
+-- Maintainer : nzeh@cs.dal.ca
+-- Stability : unstable
+-- Portability : unportable
+--
+-- Two layouts: one is a variant of the Grid layout that allows the
+-- desired aspect ratio of windows to be specified. The other is like
+-- Tall but places a grid with fixed number of rows and columns in the
+-- master area and uses an aspect-ratio-specified layout for the
+-- slaves.
+----------------------------------------------------------------------
+
+module XMonad.Layout.GridVariants ( -- * Usage
+ -- $usage
+ ChangeMasterGeom(..)
+ , Grid(..)
+ , TallGrid(..)
+ ) where
+
+import Control.Monad
+import XMonad
+import qualified XMonad.StackSet as W
+
+-- $usage
+-- This module can be used as follows:
+--
+-- > import XMonad.Layout.Master
+--
+-- Then add something like this to your layouts:
+--
+-- > Grid (16/10)
+--
+-- for a 16:10 aspect ratio grid, or
+--
+-- > TallGrid 2 3 (2/3) (16/10) (5/100)
+--
+-- for a layout with a 2x3 master grid that uses 2/3 of the screen,
+-- and a 16:10 aspect ratio slave grid. The last parameter is again
+-- the percentage by which the split between master and slave area
+-- changes in response to Expand/Shrink messages.
+--
+-- To be able to change the geometry of the master grid, add something
+-- like this to your keybindings:
+--
+-- > ((modMask .|. shiftMask, xK_equal), sendMessage $ IncMasterCols 1),
+-- > ((modMask .|. shiftMask, xK_minus), sendMessage $ IncMasterCols (-1)),
+-- > ((modMask .|. ctrlMask, xK_equal), sendMessage $ IncMasterRows 1),
+-- > ((modMask .|. ctrlMask, xK_minus), sendMessage $ IncMasterRows (-1))
+
+-- | Grid layout. The parameter is the desired x:y aspect ratio of windows
+data Grid a = Grid !Rational
+ deriving (Read, Show)
+
+instance LayoutClass Grid a where
+
+ pureLayout (Grid aspect) rect st = zip wins rects
+ where
+ wins = W.integrate st
+ nwins = length wins
+ rects = arrangeAspectGrid rect nwins aspect
+
+ description _ = "Grid"
+
+-- | TallGrid layout. Parameters are
+--
+-- - number of master rows
+-- - number of master columns
+-- - portion of screen used for master grid
+-- - x:y aspect ratio of slave windows
+-- - increment for resize messages
+data TallGrid a = TallGrid !Int !Int !Rational !Rational !Rational
+ deriving (Read, Show)
+
+instance LayoutClass TallGrid a where
+
+ pureLayout (TallGrid mrows mcols mfrac saspect _) rect st = zip wins rects
+ where
+ wins = W.integrate st
+ nwins = length wins
+ rects = arrangeTallGrid rect nwins mrows mcols mfrac saspect
+
+ pureMessage layout msg =
+ msum [ fmap (resizeMaster layout) (fromMessage msg)
+ , fmap (changeMasterGrid layout) (fromMessage msg) ]
+
+ description _ = "TallGrid"
+
+-- |The geometry change message understood by the master grid
+data ChangeMasterGeom
+ = IncMasterRows !Int -- ^Change the number of master rows
+ | IncMasterCols !Int -- ^Change the number of master columns
+ deriving Typeable
+
+instance Message ChangeMasterGeom
+
+arrangeTallGrid :: Rectangle -> Int -> Int -> Int -> Rational -> Rational -> [Rectangle]
+arrangeTallGrid rect@(Rectangle rx ry rw rh) nwins mrows mcols mfrac saspect
+ | nwins <= mwins = arrangeMasterGrid rect nwins mcols
+ | mwins == 0 = arrangeAspectGrid rect nwins saspect
+ | otherwise = (arrangeMasterGrid mrect mwins mcols) ++
+ (arrangeAspectGrid srect swins saspect)
+ where
+ mwins = mrows * mcols
+ swins = nwins - mwins
+ mrect = Rectangle rx ry rw mh
+ srect = Rectangle rx (fromIntegral ry + fromIntegral mh) rw sh
+ mh = ceiling (fromIntegral rh * mfrac)
+ sh = rh - mh
+
+arrangeMasterGrid :: Rectangle -> Int -> Int -> [Rectangle]
+arrangeMasterGrid rect nwins mcols = arrangeGrid rect nwins (min nwins mcols)
+
+arrangeAspectGrid :: Rectangle -> Int -> Rational -> [Rectangle]
+arrangeAspectGrid rect@(Rectangle _ _ rw rh) nwins aspect =
+ arrangeGrid rect nwins (min nwins ncols)
+ where
+ ncols = ceiling $ sqrt $ ( fromRational
+ ( (fromIntegral rw * fromIntegral nwins) / (fromIntegral rh * aspect) ) :: Double)
+
+arrangeGrid :: Rectangle -> Int -> Int -> [Rectangle]
+arrangeGrid (Rectangle rx ry rw rh) nwins ncols =
+ [Rectangle (fromIntegral x + rx) (fromIntegral y + ry) (fromIntegral w) (fromIntegral h)
+ | (x, y, w, h) <- rects]
+ where
+ nrows_in_cols = listDifference $ splitEvenly nwins ncols
+ x_slabs = splitIntoSlabs (fromIntegral rw) ncols
+ y_slabs = [splitIntoSlabs (fromIntegral rh) nrows | nrows <- nrows_in_cols]
+ rects_in_cols = [[(x, y, w, h) | (y, h) <- lst]
+ | ((x, w), lst) <- zip x_slabs y_slabs]
+ rects = foldr (++) [] rects_in_cols
+
+splitIntoSlabs :: Int -> Int -> [(Int, Int)]
+splitIntoSlabs width nslabs = zip (0:xs) widths
+ where
+ xs = splitEvenly width nslabs
+ widths = listDifference xs
+
+listDifference :: [Int] -> [Int]
+listDifference lst = [cur-pre | (cur,pre) <- zip lst (0:lst)]
+
+splitEvenly :: Int -> Int -> [Int]
+splitEvenly n parts = [ sz-off | (sz,off) <- zip sizes offsets]
+ where
+ size = ceiling ( (fromIntegral n / fromIntegral parts) :: Double )
+ extra = size*parts - n
+ sizes = [i*size | i <- [1..parts]]
+ offsets = (take (fromIntegral extra) [1..]) ++ [extra,extra..]
+
+resizeMaster :: TallGrid a -> Resize -> TallGrid a
+resizeMaster (TallGrid mrows mcols mfrac saspect delta) Shrink =
+ TallGrid mrows mcols (max 0 (mfrac - delta)) saspect delta
+resizeMaster (TallGrid mrows mcols mfrac saspect delta) Expand =
+ TallGrid mrows mcols (min 1 (mfrac + delta)) saspect delta
+
+changeMasterGrid :: TallGrid a -> ChangeMasterGeom -> TallGrid a
+changeMasterGrid (TallGrid mrows mcols mfrac saspect delta) (IncMasterRows d) =
+ TallGrid (max 0 (mrows + d)) mcols mfrac saspect delta
+changeMasterGrid (TallGrid mrows mcols mfrac saspect delta) (IncMasterCols d) =
+ TallGrid mrows (max 0 (mcols + d)) mfrac saspect delta
diff --git a/xmonad-contrib.cabal b/xmonad-contrib.cabal
index 3f760fd..e083df3 100644
--- a/xmonad-contrib.cabal
+++ b/xmonad-contrib.cabal
@@ -132,6 +132,7 @@ library
XMonad.Layout.FixedColumn
XMonad.Layout.Gaps
XMonad.Layout.Grid
+ XMonad.Layout.GridVariants
XMonad.Layout.HintedGrid
XMonad.Layout.HintedTile
XMonad.Layout.IM