aboutsummaryrefslogblamecommitdiffstats
path: root/Operations.hs
blob: ad8eaf0c9a435b6bdac9694fcb9668a54a7a85c7 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16















                               
                 


                              
 






                                                                        

                             
                              
                          
                              

                                                                      

                                                 
                                                        
                              


                                                                         
                                                                          



                                                                  












                                                                                 
                                                                     
                                                                        
                                                   

















                                                                             
                 




                                                          














                                                                          

                                 
                                                                  
                     
                    





                                                                      
           


                                                                    






                                                                


                                                                   

                    
                                                                               




                                           

                                                 
                       



                                                                            
           
 
                                                              
                        
                                            













                                                                                 

















                                                                              
                                                                             
                         


                                           








                                                          
 





























                                                                                          






                                              

                                                          
               
                           
                            
 





                                                          
                                                                    
                         

                                                     

                                                                           
                                                                 








                                                             



                                                              




                                                              




                                                               
               
                                                                  





























                                                                    
                           




                                                               
               
                                          
                             


























                                                                                      



                                      










                                                                                                                       
module Operations where

import Data.List
import Data.Maybe
import Data.Bits
import qualified Data.Map as M

import Control.Monad.State

import System.Posix.Process
import System.Environment

import Graphics.X11.Xlib
import Graphics.X11.Xlib.Extras

import XMonad
import Data.Ratio

import qualified StackSet as W


-- ---------------------------------------------------------------------
-- Managing windows

-- | refresh. Refresh the currently focused window. Resizes to full
-- screen and raises the window.
refresh :: X ()
refresh = do
    ws     <- gets workspace
    ws2sc  <- gets wsOnScreen
    xinesc <- gets xineScreens
    d      <- gets display
    fls    <- gets layoutDescs
    let move w (Rectangle p q r s) = io $ moveResizeWindow d w p q r s
        flipRect (Rectangle p q r s) = Rectangle q p s r
    flip mapM_ (M.assocs ws2sc) $ \(n, scn) -> do
        let sc = xinesc !! scn
            fl = M.findWithDefault basicLayoutDesc n fls
            l  = layoutType fl
            fullWindow w = move w sc >> io (raiseWindow d w)

            -- runRects draws the windows, figuring out their rectangles.
            -- The code here is for a horizontal split, and tr is possibly
            -- used to convert to the vertical case.  The comments
            -- speak in terms of the horizontal case.
            runRects :: Rectangle -> (Rectangle -> Rectangle)
                     -> (Rational -> Disposition -> Disposition)
                     -> (Disposition -> Rational) -> Rational -> [Window] -> X ()
            runRects _ _ _ _ _ [] = return () -- impossible
            runRects (Rectangle sx sy sw sh) tr dfix fracFn tf (w:s) = do
              -- get the dispositions in the relevant direction (vert/horz)
              -- as specified by fracFn.
              ds <- mapM (liftM fracFn . gets . disposition) s

              -- do some math.
              let lw = round (fromIntegral sw * tf) -- lhs width
                  rw = sw - fromIntegral lw -- rhs width
                  ns = map (/ sum ds) ds -- normalized ratios for rhs.

              -- Normalize dispositions while we have the opportunity.
              -- This is bad.  Rational numbers might space leak each
              -- time we make an adjustment.  Floating point numbers are
              -- better here.  I am being paranoid.
              zipWithM_ (\ratio a -> disposeW a (dfix ratio)) ns s

              -- do some more math.
              let ps = map (round . (* fromIntegral sh)) . scanl (+) 0 $ ns
                  -- ps are the vertical positions, [p1 = 0, p2, ..., pn, sh]
                  xs = map fromIntegral . zipWith (-) (tail ps) $ ps
                  -- xs are the heights of windows, [p2-p1,p3-p2,...,sh-pn]
                  rects = zipWith (\p q -> Rectangle (sx + lw) p rw q) ps xs
                  -- rects are the rectangles of our windows.

              -- Move our lhs window, the big main one.
              move w (tr (Rectangle sx sy (fromIntegral lw) sh))

              -- Move our rhs windows.
              zipWithM_ (\r a -> move a (tr r)) rects s

              -- And raise this one, for good measure.
              whenJust (W.peek ws) (io . raiseWindow d)
        case l of
            Full -> whenJust (W.peekStack n ws) fullWindow
            _ -> case W.index n ws of
                []  -> return ()
                [w] -> fullWindow w
                s   -> case l of
                         Horz -> (runRects sc
                                           id
                                           (\r dp -> dp {horzFrac = r})
                                           horzFrac
                                           (horzTileFrac fl)
                                           s)
                         Vert -> (runRects (flipRect sc)
                                           flipRect
                                           (\r dp -> dp {vertFrac = r})
                                           vertFrac
                                           (vertTileFrac fl)
                                           s)
                         _    -> error "Operations.refresh: the absurdly \
                                       \impossible happened.  Please \
                                       \complain about this."
    whenJust (W.peek ws) setFocus

-- | switchLayout.  Switch to another layout scheme.  Switches the
-- current workspace.
switchLayout :: X ()
switchLayout = layout $ \fl -> fl { layoutType = rot (layoutType fl) }

-- | changeVert.  Changes the vertical split, if it's visible.
changeVert :: Rational -> X ()
changeVert delta = do
  l <- gets (layoutType . currentDesc)
  case l of
    Vert -> layout $ \d -> d {vertTileFrac = min 1 $
                                             max 0 $
                                             vertTileFrac d + delta}
    _    -> return ()

-- | changeHorz.  Changes the horizontal split, if it's visible.
changeHorz :: Rational -> X ()
changeHorz delta = do
  l <- gets (layoutType . currentDesc)
  case l of
   Horz -> layout $ \d -> d {horzTileFrac = min 1 $
                                            max 0 $
                                            horzTileFrac d + delta}
   _    -> return ()

-- | changeSize.  Changes the size of the window, except in Full mode, with the
-- size remaining above the given mini-mum.
changeSize :: Rational -> Rational -> X ()
changeSize delta mini = do
  l <- gets (layoutType . currentDesc)
  mw <- gets (W.peek . workspace)
  whenJust mw $ \w -> do  -- This is always Just.
    case l of
      Full -> return ()
      Horz -> disposeW w $ \d -> d {horzFrac = max mini $
                                               horzFrac d + delta}
      Vert -> disposeW w $ \d -> d {vertFrac = max mini $
                                               vertFrac d + delta} -- hrm...
    refresh

-- | layout. Modify the current workspace's layout with a pure
-- function and refresh.
layout :: (LayoutDesc -> LayoutDesc) -> X ()
layout f = do
    modify $ \s ->
        let n  = W.current . workspace $ s
            fl = currentDesc s
        in s { layoutDescs = M.insert n (f fl) (layoutDescs s) }
    refresh

-- | disposeW. Changes the disposition of a particular window.
disposeW :: Window -> (Disposition -> Disposition) -> X ()
disposeW w f = modify $ \s -> let d = f (disposition w s)
                              in s {dispositions = M.insert w d (dispositions s)}
               -- NO refresh.  Do not put refresh here.
               -- refresh calls this function.


-- | windows. Modify the current window list with a pure function, and refresh
windows :: (WorkSpace -> WorkSpace) -> X ()
windows f = do
    modify $ \s -> s { workspace = f (workspace s) }
    refresh
    ws <- gets workspace
    trace (show ws) -- log state changes to stderr

-- | hide. Hide a window by moving it offscreen.
hide :: Window -> X ()
hide w = withDisplay $ \d -> do
    (sw,sh) <- gets dimensions
    io $ moveWindow d w (2*fromIntegral sw) (2*fromIntegral sh)

-- ---------------------------------------------------------------------
-- Window operations

-- | setButtonGrab. Tell whether or not to intercept clicks on a given window
buttonsToGrab :: [Button]
buttonsToGrab = [button1, button2, button3]

setButtonGrab :: Bool -> Window -> X ()
setButtonGrab True  w = withDisplay $ \d -> io $
    flip mapM_ buttonsToGrab $ \b ->
        grabButton d b anyModifier w False
                   (buttonPressMask .|. buttonReleaseMask)
                   grabModeAsync grabModeSync none none

setButtonGrab False w = withDisplay $ \d -> io $
    flip mapM_ buttonsToGrab $ \b ->
        ungrabButton d b anyModifier w

-- | manage. Add a new window to be managed in the current workspace. Bring it into focus.
-- If the window is already under management, it is just raised.
--
-- When we start to manage a window, it gains focus.
--
manage :: Window -> X ()
manage w = do
    withDisplay $ \d -> io $ do
        selectInput d w $ structureNotifyMask .|. enterWindowMask .|. propertyChangeMask
        mapWindow d w
    setFocus w
    windows $ W.push w

-- | unmanage. A window no longer exists, remove it from the window
-- list, on whatever workspace it is.
unmanage :: Window -> X ()
unmanage w = do
    windows $ W.delete w
    withServerX $ do
      setTopFocus
      withDisplay $ \d -> io (sync d False)
      -- TODO, everything operates on the current display, so wrap it up.

-- | Grab the X server (lock it) from the X monad
withServerX :: X () -> X ()
withServerX f = withDisplay $ \dpy -> do
    io $ grabServer dpy
    f
    io $ ungrabServer dpy

safeFocus :: Window -> X ()
safeFocus w = do ws <- gets workspace
                 if W.member w ws
                    then setFocus w
                    else do b <- isRoot w
                            when b setTopFocus

-- | Explicitly set the keyboard focus to the given window
setFocus :: Window -> X ()
setFocus w = do
    ws    <- gets workspace
    ws2sc <- gets wsOnScreen

    -- clear mouse button grab and border on other windows
    flip mapM_ (M.keys ws2sc) $ \n -> do
        flip mapM_ (W.index n ws) $ \otherw -> do
            setButtonGrab True otherw
            setBorder otherw 0xdddddd

    withDisplay $ \d -> io $ setInputFocus d w revertToPointerRoot 0
    setButtonGrab False w
    setBorder w 0xff0000    -- make this configurable

    -- This does not use 'windows' intentionally.  'windows' calls refresh,
    -- which means infinite loops.
    modify $ \s -> s { workspace = W.raiseFocus w (workspace s) }

-- | Set the focus to the window on top of the stack, or root
setTopFocus :: X ()
setTopFocus = do
    ws <- gets workspace
    case W.peek ws of
        Just new -> setFocus new
        Nothing  -> gets theRoot >>= setFocus

-- | Set the border color for a particular window.
setBorder :: Window -> Pixel -> X ()
setBorder w p = withDisplay $ \d -> io $ setWindowBorder d w p

-- | raise. focus to window at offset 'n' in list.
-- The currently focused window is always the head of the list
raise :: Ordering -> X ()
raise = windows . W.rotate

-- | promote.  Make the focused window the master window in its
-- workspace , in non-fullscreen mode.
--
-- TODO: generic cycling clockwise and anticlockwise
--
promote :: X ()
promote = windows $ \w -> maybe w (\k -> W.promote k w) (W.peek w)

-- | Kill the currently focused client
kill :: X ()
kill = withDisplay $ \d -> do
    ws <- gets workspace
    whenJust (W.peek ws) $ \w -> do
        protocols <- io $ getWMProtocols d w
        wmdelt    <- gets wmdelete
        wmprot    <- gets wmprotocols
        if wmdelt `elem` protocols
            then io $ allocaXEvent $ \ev -> do
                    setEventType ev clientMessage
                    setClientMessageEvent ev w wmprot 32 wmdelt 0
                    sendEvent d w False noEventMask ev
            else io (killClient d w) >> return ()

-- | tag. Move a window to a new workspace
tag :: Int -> X ()
tag o = do
    ws <- gets workspace
    let m = W.current ws
    when (n /= m) $
        whenJust (W.peek ws) $ \w -> do
            hide w
            windows $ W.shift n
    where n = o-1

-- | view. Change the current workspace to workspce at offset 'n-1'.
view :: Int -> X ()
view o = do
    ws    <- gets workspace
    ws2sc <- gets wsOnScreen
    let m = W.current ws
    -- is the workspace we want to switch to currently visible?
    if M.member n ws2sc
        then windows $ W.view n
        else do
            sc <- case M.lookup m ws2sc of
                Nothing -> do
                    trace "Current workspace isn't visible! This should never happen!"
                    -- we don't know what screen to use, just use the first one.
                    return 0
                Just sc -> return sc
            modify $ \s -> s { wsOnScreen = M.insert n sc (M.filter (/=sc) ws2sc) }
            gets wsOnScreen >>= trace . show
            windows $ W.view n
            mapM_ hide (W.index m ws)
    setTopFocus
    where n = o-1

-- | True if window is under management by us
isClient :: Window -> X Bool
isClient w = liftM (W.member w) (gets workspace)

-- | screenWS. Returns the workspace currently visible on screen n
screenWS :: Int -> X Int
screenWS n = do
    ws2sc <- gets wsOnScreen
    -- FIXME: It's ugly to have to query this way. We need a different way to
    -- keep track of screen <-> workspace mappings.
    let ws = fmap fst $ find (\(_, scn) -> scn == (n-1)) (M.assocs ws2sc)
    return $ (fromMaybe 0 ws) + 1

-- | Restart xmonad by exec()'ing self. This doesn't save state and xmonad has
-- to be in PATH for this to work.
restart :: IO ()
restart = do
    prog <- getProgName
    args <- getArgs
    executeFile prog True args Nothing

-- | Starts dmenu on the current screen. (Requires patches to dmenu for the -x
-- and -w options.)
dmenu :: X ()
dmenu = do
    xinesc <- gets xineScreens
    ws     <- gets workspace
    ws2sc  <- gets wsOnScreen
    let curscreen = fromMaybe 0 (M.lookup (W.current ws) ws2sc)
        sc = xinesc !! curscreen
    spawn $ "exe=`dmenu_path | dmenu -x " ++ (show $ rect_x sc) ++ " -w " ++ (show $ rect_width sc) ++ "` && exec $exe"