aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--XMonad/Hooks/ManageDocks.hs145
1 files changed, 61 insertions, 84 deletions
diff --git a/XMonad/Hooks/ManageDocks.hs b/XMonad/Hooks/ManageDocks.hs
index 809f7e1..e215b49 100644
--- a/XMonad/Hooks/ManageDocks.hs
+++ b/XMonad/Hooks/ManageDocks.hs
@@ -11,83 +11,52 @@
-- Stability : unstable
-- Portability : unportable
--
--- Makes xmonad detect windows with type DOCK and does not put them in
--- layouts. It also detects window with STRUT set and modifies the
--- gap accordingly.
---
--- It also allows you to reset the gap to reflect the state of current STRUT
--- windows (for example, after you resized or closed a panel), and to toggle the Gap
--- in a STRUT-aware fashion.
---
--- The avoidStruts layout modifier allows you to make xmonad dynamically
--- avoid overlapping windows with panels. You can (optionally) enable this
--- on a selective basis, so that some layouts will effectively hide the
--- panel, by placing windows on top of it. An example use of avoidStruts
--- would be:
---
--- > layoutHook = Layout $ toggleLayouts (noBorders Full) $ avoidStruts $
--- > your actual layouts here ||| ...
---
--- You may also wish to bind a key to sendMessage ToggleStruts, which will
--- toggle the avoidStruts behavior, so you can hide your panel at will.
---
--- This would enable a full-screen mode that overlaps the panel, while all
--- other layouts avoid the panel.
---
------------------------------------------------------------------------------
+-- This module provides tools to automatically manage 'dock' type programs,
+-- such as gnome-panel, kicker, dzen, and xmobar.
module XMonad.Hooks.ManageDocks (
-- * Usage
-- $usage
- manageDocksHook
- ,resetGap
- ,toggleGap
- ,avoidStruts, ToggleStruts(ToggleStruts)
+ manageDocks, AvoidStruts, avoidStruts, ToggleStruts(ToggleStruts)
) where
+
+-----------------------------------------------------------------------------
import XMonad
-import qualified XMonad.StackSet as W
import Foreign.C.Types (CLong)
import Data.Maybe (catMaybes)
-
-- $usage
--- Add the imports to your configuration file and add the manageHook:
+-- To use this module, add the following import to @~\/.xmonad\/xmonad.hs@:
--
-- > import XMonad.Hooks.ManageDocks
--
--- > manageHook w _ _ _ = manageDocksHook w
+-- The first component is a 'ManageHook' which recognizes these windows. To
+-- enable it:
--
--- and comment out the default `manageHook _ _ _ _ = return id` line.
+-- > manageHook = ... <+> manageDocks
--
--- Then you can bind resetGap or toggleGap as you wish:
+-- The second component is a layout modifier that prevents windows from
+-- overlapping these dock windows. It is intended to replace xmonad's
+-- so-called "gap" support. First, you must add it to your list of layouts:
+--
+-- > layoutHook = avoidStruts (tall ||| mirror tall ||| ...)
+--
+-- 'AvoidStruts' also supports toggling the dock gap, add a keybinding similar
+-- to:
+--
+-- > ,((modMask, xK_b ), sendMessage ToggleStruts)
--
--- > , ((modMask, xK_b), toggleGap)
-
--- %import XMonad.Hooks.ManageDocks
--- %def -- comment out default manageHook definition above if you uncomment this:
--- %def manageHook w _ _ _ = manageDocksHook w
--- %keybind , ((modMask, xK_b), toggleGap)
-
-- |
-- Detects if the given window is of type DOCK and if so, reveals it, but does
-- not manage it. If the window has the STRUT property set, adjust the gap accordingly.
-manageDocksHook :: Window -> X (WindowSet -> WindowSet)
-manageDocksHook w = do
- hasStrut <- getStrut w
- maybe (return ()) setGap hasStrut
-
- isDock <- checkDock w
- if isDock then do
- reveal w
- return (W.delete w)
- else do
- return id
+manageDocks :: ManageHook
+manageDocks = checkDock --> doIgnore
-- |
-- Checks if a window is a DOCK window
-checkDock :: Window -> X (Bool)
-checkDock w = do
+checkDock :: Query Bool
+checkDock = ask >>= \w -> liftX $ do
a <- getAtom "_NET_WM_WINDOW_TYPE"
d <- getAtom "_NET_WM_WINDOW_TYPE_DOCK"
mbr <- getProp a w
@@ -115,41 +84,51 @@ getProp :: Atom -> Window -> X (Maybe [CLong])
getProp a w = withDisplay $ \dpy -> io $ getWindowProperty32 dpy a w
-- |
--- Modifies the gap, setting new max
-setGap :: (Int, Int, Int, Int) -> X ()
-setGap gap = modifyGap (\_ -> max4 gap)
-
-
--- |
-- Goes through the list of windows and find the gap so that all STRUT
-- settings are satisfied.
-calcGap :: X (Int, Int, Int, Int)
+calcGap :: X Rectangle
calcGap = withDisplay $ \dpy -> do
- rootw <- asks theRoot
- -- We don’t keep track of dock like windows, so we find all of them here
- (_,_,wins) <- io $ queryTree dpy rootw
- struts <- catMaybes `fmap` mapM getStrut wins
- return $ foldl max4 (0,0,0,0) struts
-
--- |
--- Adjusts the gap to the STRUTs of all current Windows
-resetGap :: X ()
-resetGap = do
- newGap <- calcGap
- modifyGap (\_ _ -> newGap)
-
--- |
--- Removes the gap or, if already removed, sets the gap according to the windows’ STRUT
-toggleGap :: X ()
-toggleGap = do
- newGap <- calcGap
- modifyGap (\_ old -> if old == (0,0,0,0) then newGap else (0,0,0,0))
+ rootw <- asks theRoot
+ -- We don't keep track of dock like windows, so we find all of them here
+ (_,_,wins) <- io $ queryTree dpy rootw
+ struts <- catMaybes `fmap` mapM getStrut wins
+
+ -- we grab the window attributes of the root window rather than checking
+ -- the width of the screen because xlib caches this info and it tends to
+ -- be incorrect after RAndR
+ wa <- io $ getWindowAttributes dpy rootw
+ return $ reduceScreen (foldl max4 (0,0,0,0) struts)
+ $ Rectangle (fi $ wa_x wa) (fi $ wa_y wa) (fi $ wa_width wa) (fi $ wa_height wa)
-- |
-- Piecewise maximum of a 4-tuple of Ints
max4 :: (Int, Int, Int, Int) -> (Int, Int, Int, Int) -> (Int, Int, Int, Int)
max4 (a1,a2,a3,a4) (b1,b2,b3,b4) = (max a1 b1, max a2 b2, max a3 b3, max a4 b4)
+fi :: (Integral a, Num b) => a -> b
+fi = fromIntegral
+
+-- | Given strut values and the screen rectangle, compute a reduced screen
+-- rectangle.
+reduceScreen :: (Int, Int, Int, Int) -> Rectangle -> Rectangle
+reduceScreen (t, b, l, r) (Rectangle rx ry rw rh)
+ = Rectangle (rx + fi l) (ry + fi t) (rw - fi r) (rh - fi b)
+
+r2c :: Rectangle -> (Position, Position, Position, Position)
+r2c (Rectangle x y w h) = (x, y, x + fi w, y + fi h)
+
+c2r :: (Position, Position, Position, Position) -> Rectangle
+c2r (x1, y1, x2, y2) = Rectangle x1 y1 (fi $ x2 - x1) (fi $ y2 - y1)
+
+-- | Given a bounding rectangle 's' and another rectangle 'r', compute a
+-- rectangle 'r' that fits inside 's'.
+fitRect :: Rectangle -> Rectangle -> Rectangle
+fitRect s r
+ = c2r (max sx1 rx1, max sy1 ry1, min sx2 rx2, min sy2 ry2)
+ where
+ (sx1, sy1, sx2, sy2) = r2c s
+ (rx1, ry1, rx2, ry2) = r2c r
+
-- | Adjust layout automagically.
avoidStruts :: LayoutClass l a => l a -> AvoidStruts l a
avoidStruts = AvoidStruts True
@@ -160,10 +139,8 @@ data ToggleStruts = ToggleStruts deriving (Read,Show,Typeable)
instance Message ToggleStruts
instance LayoutClass l a => LayoutClass (AvoidStruts l) a where
- doLayout (AvoidStruts True lo) (Rectangle x y w h) s =
- do (t,b,l,r) <- calcGap
- let rect = Rectangle (x+fromIntegral l) (y+fromIntegral t)
- (w-fromIntegral l-fromIntegral r) (h-fromIntegral t-fromIntegral b)
+ doLayout (AvoidStruts True lo) r s =
+ do rect <- fmap (flip fitRect r) calcGap
(wrs,mlo') <- doLayout lo rect s
return (wrs, AvoidStruts True `fmap` mlo')
doLayout (AvoidStruts False lo) r s = do (wrs,mlo') <- doLayout lo r s