mirror of
https://github.com/xmonad/xmonad.git
synced 2025-08-01 20:51:55 -07:00
346 lines
12 KiB
Haskell
346 lines
12 KiB
Haskell
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
|