diff options
-rw-r--r-- | XMonad/Layout/LayoutHints.hs | 173 |
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 |