diff options
-rw-r--r-- | XMonad/Layout/Monitor.hs | 131 | ||||
-rw-r--r-- | xmonad-contrib.cabal | 1 |
2 files changed, 132 insertions, 0 deletions
diff --git a/XMonad/Layout/Monitor.hs b/XMonad/Layout/Monitor.hs new file mode 100644 index 0000000..d46f914 --- /dev/null +++ b/XMonad/Layout/Monitor.hs @@ -0,0 +1,131 @@ +{-# LANGUAGE TypeSynonymInstances, MultiParamTypeClasses, DeriveDataTypeable, PatternGuards #-} + +----------------------------------------------------------------------------- +-- | +-- Module : XMonad.Layout.Monitor +-- Copyright : (c) Roman Cheplyaka +-- License : BSD-style (see LICENSE) +-- +-- Maintainer : Roman Cheplyaka <roma@ro-che.info> +-- Stability : unstable +-- Portability : unportable +-- +-- Layout modfier for displaying some window (monitor) above other windows +-- +----------------------------------------------------------------------------- +module XMonad.Layout.Monitor ( + -- * Usage + -- $usage + + -- * Hints + -- $hints + + -- * TODO + -- $todo + Property(..), + MonitorMessage(..), + addMonitor, + addPersistentMonitor, + addNamedMonitor, + addNamedPersistentMonitor + ) where + +import XMonad +import XMonad.Layout.LayoutModifier +import XMonad.Util.WindowProperties +import Control.Monad + +-- $usage +-- You can use this module with the following in your @~\/.xmonad\/xmonad.hs@: +-- +-- > import XMonad.Layout.Monitor +-- +-- Then add monitor to desired layouts: +-- +-- > myLayouts = addMonitor (ClassName "Cairo-clock" `And` Title "MacSlow's Cairo-Clock") (Rectangle (1280-150) (800-150) 150 150) $ tall ||| Full ||| ... +-- +-- After that, if there exists a window with specified properties, it will be +-- displayed on top of all /tiled/ (not floated) windows on specified +-- position. +-- +-- It's also useful to add some keybinding to toggle monitor visibility: +-- +-- > , ((mod1Mask, xK_u ), sendMessage ToggleMonitor) +-- +-- Screenshot: <http://www.haskell.org/haskellwiki/Image:Xmonad-clock.png> + +data Monitor a = Monitor { + prop :: Property, -- a window which satisfies that property is chosen as monitor + rect :: Rectangle, -- where to put monitor + visible :: Bool, -- is it visible? + mbName :: (Maybe String), -- name of monitor (useful when we have many of them) + persistent :: Bool -- on all layouts? + } deriving (Read, Show) + +data MonitorMessage = ToggleMonitor | ShowMonitor | HideMonitor + | ToggleMonitorNamed String + | ShowMonitorNamed String + | HideMonitorNamed String + deriving (Read,Show,Eq,Typeable) +instance Message MonitorMessage + +withMonitor :: Property -> a -> (Window -> X a) -> X a +withMonitor p a fn = do + monitorWindows <- allWithProperty p + case monitorWindows of + [] -> return a + w:_ -> fn w + +instance LayoutModifier Monitor Window where + redoLayout mon _ _ rects = withMonitor (prop mon) (rects, Nothing) $ \w -> + if visible mon + then do tileWindow w (rect mon) + reveal w + return ((w,rect mon):rects, Nothing) + else do hide w + return (rects, Nothing) + handleMess mon mess + | Just ToggleMonitor <- fromMessage mess = return $ Just $ mon { visible = not $ visible mon } + | Just (ToggleMonitorNamed n) <- fromMessage mess = return $ + if mbName mon `elem` [Just n, Nothing] then Just $ mon { visible = not $ visible mon } else Nothing + | Just ShowMonitor <- fromMessage mess = return $ Just $ mon { visible = True } + | Just (ShowMonitorNamed n) <- fromMessage mess = return $ + if mbName mon `elem` [Just n, Nothing] then Just $ mon { visible = True } else Nothing + | Just HideMonitor <- fromMessage mess = return $ Just $ mon { visible = False } + | Just (HideMonitorNamed n) <- fromMessage mess = return $ + if mbName mon `elem` [Just n, Nothing] then Just $ mon { visible = False } else Nothing + | Just Hide <- fromMessage mess = do unless (persistent mon) $ withMonitor (prop mon) () hide; return Nothing + | otherwise = return Nothing + +addMonitor :: Property -> Rectangle -> l a -> ModifiedLayout Monitor l a +addMonitor p r = ModifiedLayout (Monitor p r True Nothing False) +addPersistentMonitor :: Property -> Rectangle -> l a -> ModifiedLayout Monitor l a +addPersistentMonitor p r = ModifiedLayout (Monitor p r True Nothing True) +addNamedMonitor :: String -> Property -> Rectangle -> l a -> ModifiedLayout Monitor l a +addNamedMonitor name p r = ModifiedLayout (Monitor p r True (Just name) False) +addNamedPersistentMonitor :: String -> Property -> Rectangle -> l a -> ModifiedLayout Monitor l a +addNamedPersistentMonitor name p r = ModifiedLayout (Monitor p r True (Just name) True) + +-- $hints +-- - This module assumes that there is only one window satisfying property exists. Also it's good idea to make it unmanaged and (optionally) hide it using ManageHook: +-- +-- > className =? "Cairo-clock"--> (ask >>= \w -> liftX (hide w) >> doF (W.delete w)) +-- +-- - If you want monitor to be available on /all/ layouts, there's no point in +-- hiding it. Also use 'addPersistentMonitor' instead of +-- 'addMonitor' to avoid unnecessary flickering. You can still toggle +-- monitor with a keybinding. +-- +-- - You can use several monitors with nested modifiers. Give them a name using +-- 'addNamedMonitor' or 'addNamedPersistentMonitor' to be able to toggle +-- them independently. +-- +-- - You can display monitor only on specific workspaces with +-- "XMonad.Layout.PerWorkspace". + +-- $todo +-- - make Monitor remember the window it manages +-- +-- - automatically unmanage the window? +-- +-- - specify position relative to the screen diff --git a/xmonad-contrib.cabal b/xmonad-contrib.cabal index 81afec0..5012b83 100644 --- a/xmonad-contrib.cabal +++ b/xmonad-contrib.cabal @@ -143,6 +143,7 @@ library XMonad.Layout.Magnifier XMonad.Layout.Master XMonad.Layout.Maximize + XMonad.Layout.Monitor XMonad.Layout.MosaicAlt XMonad.Layout.MultiToggle XMonad.Layout.MultiToggle.Instances |