aboutsummaryrefslogtreecommitdiffstats
path: root/XMonad/Layout
diff options
context:
space:
mode:
Diffstat (limited to 'XMonad/Layout')
-rw-r--r--XMonad/Layout/FixedColumn.hs91
1 files changed, 91 insertions, 0 deletions
diff --git a/XMonad/Layout/FixedColumn.hs b/XMonad/Layout/FixedColumn.hs
new file mode 100644
index 0000000..912a92d
--- /dev/null
+++ b/XMonad/Layout/FixedColumn.hs
@@ -0,0 +1,91 @@
+{-# LANGUAGE FlexibleInstances, MultiParamTypeClasses, TypeSynonymInstances #-}
+-----------------------------------------------------------------------------
+-- |
+-- Module : XMonad.Layout.FixedColumn
+-- Copyright : (c) 2008 Justin Bogner <mail@justinbogner.com>
+-- License : BSD3-style (as xmonad)
+--
+-- Maintainer : Justin Bogner <mail@justinbogner.com>
+-- Stability : unstable
+-- Portability : unportable
+--
+-- A layout much like Tall, but using a multiple of a window's minimum
+-- resize amount instead of a percentage of screen to decide where to
+-- split. This is useful when you usually leave a text editor or
+-- terminal in the master pane and like it to be 80 columns wide.
+--
+-----------------------------------------------------------------------------
+
+module XMonad.Layout.FixedColumn (
+ -- * Usage
+ -- $usage
+ FixedColumn(..)
+) where
+
+import Control.Monad (msum)
+import Data.Maybe (fromMaybe)
+import Graphics.X11.Xlib (Window, rect_width)
+import Graphics.X11.Xlib.Extras ( getWMNormalHints
+ , getWindowAttributes
+ , sh_base_size
+ , sh_resize_inc
+ , wa_border_width)
+
+import XMonad.Core (X, LayoutClass(..), fromMessage, io, withDisplay)
+import XMonad.Layout (Resize(..), IncMasterN(..), tile)
+import XMonad.StackSet as W
+
+-- $usage
+-- You can use this module with the following in your @~\/.xmonad\/xmonad.hs@:
+--
+-- > import XMonad.Layout.FixedColumn
+--
+-- Then edit your @layoutHook@ by adding the FixedColumn layout:
+--
+-- > myLayouts = FixedColumn 20 80 10 ||| Full ||| etc..
+-- > main = xmonad defaultConfig { layoutHook = myLayouts }
+--
+-- For more detailed instructions on editing the layoutHook see:
+--
+-- "XMonad.Doc.Extending#Editing_the_layout_hook"
+
+-- | A tiling mode based on preserving a nice fixed width
+-- window. Supports 'Shrink', 'Expand' and 'IncMasterN'.
+data FixedColumn a = FixedColumn !Int -- Number of windows in the master pane
+ !Int -- Number to increment by when resizing
+ !Int -- Default width of master pane
+ !Int -- Column width for normal windows
+ deriving (Read, Show)
+
+instance LayoutClass FixedColumn Window where
+ doLayout (FixedColumn nmaster _ ncol fallback) r s = do
+ fws <- mapM (widthCols fallback ncol) ws
+ let frac = maximum (take nmaster fws) // rect_width r
+ rs = tile frac r nmaster (length ws)
+ return $ (zip ws rs, Nothing)
+ where ws = W.integrate s
+ x // y = fromIntegral x / fromIntegral y
+
+ pureMessage (FixedColumn nmaster delta ncol fallback) m =
+ msum [fmap resize (fromMessage m)
+ ,fmap incmastern (fromMessage m)]
+ where resize Shrink
+ = FixedColumn nmaster delta (max 0 $ ncol - delta) fallback
+ resize Expand
+ = FixedColumn nmaster delta (ncol + delta) fallback
+ incmastern (IncMasterN d)
+ = FixedColumn (max 0 (nmaster+d)) delta ncol fallback
+
+ description _ = "FixedColumn"
+
+-- | Determine the width of @w@ given that we would like it to be @n@
+-- columns wide, using @inc@ as a resize increment for windows that
+-- don't have one
+widthCols :: Int -> Int -> Window -> X Int
+widthCols inc n w = withDisplay $ \d -> io $ do
+ sh <- getWMNormalHints d w
+ bw <- fmap (fromIntegral . wa_border_width) $ getWindowAttributes d w
+ let widthHint f = f sh >>= return . fromIntegral . fst
+ oneCol = fromMaybe inc $ widthHint sh_resize_inc
+ base = fromMaybe 0 $ widthHint sh_base_size
+ return $ 2 * bw + base + n * oneCol