aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--XMonad/Layout/LayoutHints.hs173
1 files changed, 162 insertions, 11 deletions
diff --git a/XMonad/Layout/LayoutHints.hs b/XMonad/Layout/LayoutHints.hs
index c952cf2..7419573 100644
--- a/XMonad/Layout/LayoutHints.hs
+++ b/XMonad/Layout/LayoutHints.hs
@@ -1,5 +1,5 @@
{-# LANGUAGE FlexibleInstances, MultiParamTypeClasses, TypeSynonymInstances #-}
-
+{-# LANGUAGE ParallelListComp, PatternGuards #-}
-----------------------------------------------------------------------------
-- |
-- Module : XMonad.Layout.LayoutHints
@@ -8,7 +8,7 @@
--
-- Maintainer : none
-- Stability : unstable
--- Portability : portable
+-- Portability : unportable
--
-- Make layouts respect size hints.
-----------------------------------------------------------------------------
@@ -18,16 +18,26 @@ module XMonad.Layout.LayoutHints
-- $usage
layoutHints
, layoutHintsWithPlacement
+ , layoutHintsToCentre
, LayoutHints
- , placeRectangle
- ) where
+ ) where
+
+import XMonad(LayoutClass(runLayout), X, mkAdjust, Window,
+ Dimension, Position, Rectangle(Rectangle))
+import qualified XMonad.StackSet as W
-import XMonad hiding ( trace )
-import XMonad.Layout.LayoutModifier
-import XMonad.Layout.Decoration ( isInStack )
+import XMonad.Hooks.ManageDocks(Direction(..))
+import XMonad.Layout.Decoration(isInStack)
+import XMonad.Layout.LayoutModifier(ModifiedLayout(..),
+ LayoutModifier(modifyLayout, redoLayout, modifierDescription))
+import Control.Applicative((<$>))
+import Control.Arrow(Arrow((***), second))
+import Control.Monad(Monad(return), mapM, join)
+import Data.Function(on)
+import Data.List(sortBy)
-import Control.Applicative ( (<$>) )
-import Control.Arrow ( second )
+import Data.Set (Set)
+import qualified Data.Set as Set
-- $usage
-- You can use this module with the following in your @~\/.xmonad\/xmonad.hs@:
@@ -45,6 +55,10 @@ import Control.Arrow ( second )
-- > myLayouts = layoutHintsWithPlacement (0.5, 0.5) (Tall 1 (3/100) (1/2))
-- > ||| Full ||| etc..
--
+-- Or, to make a reasonable attempt to eliminate gaps between windows:
+--
+-- > myLayouts = layoutHintsToCentre (Tall 1 (3/100) (1/2))
+--
-- For more detailed instructions on editing the layoutHook see:
--
-- "XMonad.Doc.Extending#Editing_the_layout_hook"
@@ -61,7 +75,18 @@ layoutHintsWithPlacement :: (LayoutClass l a) => (Double, Double)
-> l a -> ModifiedLayout LayoutHints l a
layoutHintsWithPlacement rs = ModifiedLayout (LayoutHints rs)
-data LayoutHints a = LayoutHints (Double, Double)
+-- | @layoutHintsToCentre layout@ applies hints, sliding the window to the
+-- centre of the screen and expanding its neighbours to fill the gaps. Windows
+-- are never expanded in a way that increases overlap.
+--
+-- @layoutHintsToCentre@ only makes one pass at resizing the neighbours of
+-- hinted windows, so with some layouts (ex. the arrangment with two 'Mirror'
+-- 'Tall' stacked vertically), @layoutHintsToCentre@ may leave some gaps.
+-- Simple layouts like 'Tall' are unaffected.
+layoutHintsToCentre :: (LayoutClass l a) => l a -> ModifiedLayout LayoutHintsToCentre l a
+layoutHintsToCentre = ModifiedLayout LayoutHintsToCentre
+
+data LayoutHints a = LayoutHints (Double, Double)
deriving (Read, Show)
instance LayoutModifier LayoutHints Window where
@@ -83,4 +108,130 @@ placeRectangle :: RealFrac r => (r, r) -> Rectangle -> Rectangle -> Rectangle
placeRectangle (rx, ry) (Rectangle x0 y0 w h) (Rectangle _ _ dx dy)
= Rectangle (align x0 dx w rx) (align y0 dy h ry) dx dy
where align :: RealFrac r => Position -> Dimension -> Dimension -> r -> Position
- align z0 dz d r = z0 + truncate (fromIntegral (d - dz) * r) \ No newline at end of file
+ align z0 dz d r = z0 + truncate (fromIntegral (d - dz) * r)
+
+fitting :: [Rectangle] -> Int
+fitting rects = sum $ do
+ r <- rects
+ return $ length $ filter (touching r) rects
+
+applyOrder :: Rectangle -> [(Window, Rectangle)] -> [[(Window, Rectangle)]]
+applyOrder root wrs = do
+ -- perhaps it would just be better to take all permutations, or apply the
+ -- resizing multiple times
+ f <- [maximum, minimum, sum, sum . map sq]
+ return $ sortBy (compare `on` (f . distance)) wrs
+ where distFC = uncurry ((+) `on` sq) . pairWise (-) (centre root)
+ distance = map distFC . corners . snd
+ pairWise f (a,b) (c,d) = (f a c, f b d)
+ sq = join (*)
+
+data LayoutHintsToCentre a = LayoutHintsToCentre deriving (Read, Show)
+
+instance LayoutModifier LayoutHintsToCentre Window where
+ modifyLayout _ ws@(W.Workspace _ _ Nothing) r = runLayout ws r
+ modifyLayout _ ws@(W.Workspace _ _ (Just st)) r = do
+ (arrs,ol) <- runLayout ws r
+ flip (,) ol
+ . head . reverse . sortBy (compare `on` (fitting . map snd))
+ <$> mapM (applyHints st r) (applyOrder r arrs)
+
+-- apply hints to first, grow adjacent windows
+applyHints :: W.Stack Window -> Rectangle -> [(Window, Rectangle)] -> X [(Window, Rectangle)]
+applyHints _ _ [] = return []
+applyHints s root ((w,lrect@(Rectangle a b c d)):xs) = do
+ adj <- mkAdjust w
+ let (c',d') = adj (c,d)
+ redr = placeRectangle (centrePlacement root lrect :: (Double,Double)) lrect
+ $ if isInStack s w then Rectangle a b c' d' else lrect
+
+ ds = (fromIntegral c - fromIntegral c',fromIntegral d - fromIntegral d')
+ growOther' r = growOther ds lrect (freeDirs root lrect) r
+ mapSnd f = map (second f)
+ next <- applyHints s root $ mapSnd growOther' xs
+ return $ (w,redr):next
+
+growOther :: (Position, Position) -> Rectangle -> Set Direction -> Rectangle -> Rectangle
+growOther ds lrect fds r
+ | dirs <- flipDir <$> Set.toList (Set.intersection adj fds)
+ , not $ any (uncurry opposite) $ cross dirs =
+ foldr (flip grow ds) r dirs
+ | otherwise = r
+ where
+ adj = adjacent lrect r
+ cross xs = [ (a,b) | a <- xs, b <- xs ]
+
+ flipDir :: Direction -> Direction
+ flipDir d = case d of { L -> R; U -> D; R -> L; D -> U }
+
+ opposite :: Direction -> Direction -> Bool
+ opposite x y = flipDir x == y
+
+-- | Leave the opposite edges where they were
+grow :: Direction -> (Position,Position) -> Rectangle -> Rectangle
+grow L (px,_ ) (Rectangle x y w h) = Rectangle (x-px) y (w+fromIntegral px) h
+grow U (_ ,py) (Rectangle x y w h) = Rectangle x (y-py) w (h+fromIntegral py)
+grow R (px,_ ) (Rectangle x y w h) = Rectangle x y (w+fromIntegral px) h
+grow D (_ ,py) (Rectangle x y w h) = Rectangle x y w (h+fromIntegral py)
+
+comparingEdges :: ([Position] -> [Position] -> Bool) -> Rectangle -> Rectangle -> Set Direction
+comparingEdges surrounds r1 r2 = Set.fromList $ map fst $ filter snd [ (\k -> (dir,k)) $
+ any and [[dir `elem` [R,L], allEq [a,c,w,y], [b,d] `surrounds` [x,z]]
+ ,[dir `elem` [U,D], allEq [b,d,x,z], [a,c] `surrounds` [w,y]]]
+ | ((a,b),(c,d)) <- edge $ corners r1
+ | ((w,x),(y,z)) <- edge $ delay 2 $ corners r2
+ | dir <- [U,R,D,L]]
+ where edge (x:xs) = zip (x:xs) (xs ++ [x])
+ edge [] = []
+ delay n xs = drop n xs ++ take n xs
+ allEq = all (uncurry (==)) . edge
+
+-- | in what direction is the second window from the first that can expand if the
+-- first is shrunk, assuming that the root window is fully covered:
+-- one direction for a common edge
+-- two directions for a common corner
+adjacent :: Rectangle -> Rectangle -> Set Direction
+adjacent = comparingEdges (all . onClosedInterval)
+
+-- | True whenever two edges touch. not (Set.null $ adjacent x y) ==> touching x y
+touching :: Rectangle -> Rectangle -> Bool
+touching a b = not . Set.null $ comparingEdges c a b
+ where c x y = any (onClosedInterval x) y || any (onClosedInterval y) x
+
+onClosedInterval :: Ord a => [a] -> a -> Bool
+onClosedInterval bds x = minimum bds <= x && maximum bds >= x
+
+-- | starting top left going clockwise
+corners :: Rectangle -> [(Position, Position)]
+corners (Rectangle x y w h) = [(x,y)
+ ,(x+fromIntegral w, y)
+ ,(x+fromIntegral w, y+fromIntegral h)
+ ,(x, y+fromIntegral h)]
+
+centre :: Rectangle -> (Position, Position)
+centre (Rectangle x y w h) = (avg x w, avg y h)
+ where avg a b = a + fromIntegral b `div` 2
+
+centrePlacement :: RealFrac r => Rectangle -> Rectangle -> (r, r)
+centrePlacement = centrePlacement' clamp
+ where clamp n = case signum n of
+ 0 -> 0.5
+ 1 -> 1
+ _ -> 0
+
+freeDirs :: Rectangle -> Rectangle -> Set Direction
+freeDirs root = Set.fromList . uncurry (++) . (lr *** ud)
+ . centrePlacement' signum root
+ where
+ lr 1 = [L]
+ lr (-1) = [R]
+ lr _ = [L,R]
+ ud 1 = [U]
+ ud (-1) = [D]
+ ud _ = [U,D]
+
+centrePlacement' :: (Position -> r) -> Rectangle -> Rectangle -> (r, r)
+centrePlacement' cf root assigned
+ = (cf $ cx - cwx, cf $ cy - cwy)
+ where (cx,cy) = centre root
+ (cwx,cwy) = centre assigned