mirror of
https://github.com/xmonad/xmonad.git
synced 2025-08-07 23:51:52 -07:00
In order to give a better account of how focus and master interact, and how each operation affects focus, we reimplement the StackSet type as a two level nested 'Zipper'. To quote Oleg: A Zipper is essentially an `updateable' and yet pure functional cursor into a data structure. Zipper is also a delimited continuation reified as a data structure. That is, we use the Zipper as a cursor which encodes the window which is in focus. Thus our data structure tracks focus correctly by construction! We then get simple, obvious semantics for e.g. insert, in terms of how it affects focus/master. Our transient-messes-with-focus bug evaporates. 'swap' becomes trivial. By moving focus directly into the stackset, we can toss some QC properties about focus handling: it is simply impossible now for focus to go wrong. As a benefit, we get a dozen new QC properties for free, governing how master and focus operate. The encoding of focus in the data type also simplifies the focus handling in Operations: several operations affecting focus are now simply wrappers over StackSet. For the full story, please read the StackSet module, and the QC properties. Finally, we save ~40 lines with the simplified logic in Operations.hs For more info, see the blog post on the implementation, http://cgi.cse.unsw.edu.au/~dons/blog/2007/05/17#xmonad_part1b_zipper
174 lines
6.0 KiB
Haskell
174 lines
6.0 KiB
Haskell
{-# OPTIONS -fglasgow-exts #-}
|
|
-----------------------------------------------------------------------------
|
|
-- |
|
|
-- Module : XMonad.hs
|
|
-- Copyright : (c) Spencer Janssen 2007
|
|
-- License : BSD3-style (see LICENSE)
|
|
--
|
|
-- Maintainer : sjanssen@cse.unl.edu
|
|
-- Stability : unstable
|
|
-- Portability : not portable, uses cunning newtype deriving
|
|
--
|
|
-----------------------------------------------------------------------------
|
|
--
|
|
-- The X monad, a state monad transformer over IO, for the window
|
|
-- manager state, and support routines.
|
|
--
|
|
|
|
module XMonad (
|
|
X, WindowSet, WorkspaceId(..), ScreenId(..), XState(..), XConf(..), Layout(..),
|
|
Typeable, Message, SomeMessage(..), fromMessage,
|
|
runX, io, withDisplay, withWorkspace, isRoot, spawn, restart, trace, whenJust, whenX
|
|
) where
|
|
|
|
import StackSet (StackSet)
|
|
|
|
import Control.Monad.State
|
|
import Control.Monad.Reader
|
|
import System.IO
|
|
import System.Posix.Process (executeFile, forkProcess, getProcessStatus, createSession)
|
|
import System.Exit
|
|
import System.Environment
|
|
import System.Directory
|
|
import Graphics.X11.Xlib
|
|
import Data.Typeable
|
|
|
|
import qualified Data.Map as M
|
|
|
|
-- | XState, the window manager state.
|
|
-- Just the display, width, height and a window list
|
|
data XState = XState
|
|
{ workspace :: !WindowSet -- ^ workspace list
|
|
, layouts :: !(M.Map WorkspaceId (Layout, [Layout])) }
|
|
-- ^ mapping of workspaces to descriptions of their layouts
|
|
|
|
data XConf = XConf
|
|
{ display :: Display -- ^ the X11 display
|
|
|
|
, theRoot :: !Window -- ^ the root window
|
|
, wmdelete :: !Atom -- ^ window deletion atom
|
|
, wmprotocols :: !Atom -- ^ wm protocols atom
|
|
, dimensions :: !(Int,Int) -- ^ dimensions of the screen,
|
|
-- used for hiding windows
|
|
|
|
, xineScreens :: ![Rectangle] -- ^ dimensions of each screen
|
|
, normalBorder :: !Color -- ^ border color of unfocused windows
|
|
, focusedBorder :: !Color } -- ^ border color of the focused window
|
|
|
|
type WindowSet = StackSet WorkspaceId Window ScreenId
|
|
|
|
-- | Virtual workspace indicies
|
|
newtype WorkspaceId = W Int deriving (Eq,Ord,Show,Enum,Num,Integral,Real)
|
|
|
|
-- | Physical screen indicies
|
|
newtype ScreenId = S Int deriving (Eq,Ord,Show,Enum,Num,Integral,Real)
|
|
|
|
------------------------------------------------------------------------
|
|
|
|
-- | The X monad, a StateT transformer over IO encapsulating the window
|
|
-- manager state
|
|
--
|
|
-- Dynamic components may be retrieved with 'get', static components
|
|
-- with 'ask'. With newtype deriving we get readers and state monads
|
|
-- instantiated on XConf and XState automatically.
|
|
--
|
|
newtype X a = X (ReaderT XConf (StateT XState IO) a)
|
|
deriving (Functor, Monad, MonadIO, MonadState XState, MonadReader XConf)
|
|
|
|
-- | Run the X monad, given a chunk of X monad code, and an initial state
|
|
-- Return the result, and final state
|
|
runX :: XConf -> XState -> X a -> IO ()
|
|
runX c st (X a) = runStateT (runReaderT a c) st >> return ()
|
|
|
|
-- ---------------------------------------------------------------------
|
|
-- Convenient wrappers to state
|
|
|
|
-- | Run a monad action with the current display settings
|
|
withDisplay :: (Display -> X ()) -> X ()
|
|
withDisplay f = asks display >>= f
|
|
|
|
-- | Run a monadic action with the current workspace
|
|
withWorkspace :: (WindowSet -> X a) -> X a
|
|
withWorkspace f = gets workspace >>= f
|
|
|
|
-- | True if the given window is the root window
|
|
isRoot :: Window -> X Bool
|
|
isRoot w = liftM (w==) (asks theRoot)
|
|
|
|
------------------------------------------------------------------------
|
|
-- Layout handling
|
|
|
|
-- | The different layout modes
|
|
-- 'doLayout', a pure function to layout a Window set 'modifyLayout',
|
|
-- 'modifyLayout' can be considered a branch of an exception handler.
|
|
--
|
|
data Layout = Layout { doLayout :: Rectangle -> [Window] -> X [(Window, Rectangle)]
|
|
, modifyLayout :: SomeMessage -> Maybe Layout }
|
|
|
|
-- Based on ideas in /An Extensible Dynamically-Typed Hierarchy of Exceptions/,
|
|
-- Simon Marlow, 2006. Use extensible messages to the modifyLayout handler.
|
|
--
|
|
-- User-extensible messages must be a member of this class:
|
|
--
|
|
class Typeable a => Message a
|
|
|
|
--
|
|
-- A wrapped value of some type in the Message class.
|
|
--
|
|
data SomeMessage = forall a. Message a => SomeMessage a
|
|
|
|
--
|
|
-- And now, unwrap a given, unknown Message type, performing a (dynamic)
|
|
-- type check on the result.
|
|
--
|
|
fromMessage :: Message m => SomeMessage -> Maybe m
|
|
fromMessage (SomeMessage m) = cast m
|
|
|
|
-- ---------------------------------------------------------------------
|
|
-- General utilities
|
|
|
|
-- | Lift an IO action into the X monad
|
|
io :: IO a -> X a
|
|
io = liftIO
|
|
|
|
-- | spawn. Launch an external application
|
|
spawn :: String -> X ()
|
|
spawn x = io $ do
|
|
pid <- forkProcess $ do
|
|
forkProcess (createSession >> executeFile "/bin/sh" False ["-c", x] Nothing)
|
|
exitWith ExitSuccess
|
|
return ()
|
|
getProcessStatus True False pid
|
|
return ()
|
|
|
|
-- | 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
|
|
prog_path <- findExecutable prog
|
|
case prog_path of
|
|
Nothing -> return () -- silently fail
|
|
Just p -> do args <- getArgs
|
|
executeFile p True args Nothing
|
|
|
|
-- | Run a side effecting action with the current workspace. Like 'when' but
|
|
whenJust :: Maybe a -> (a -> X ()) -> X ()
|
|
whenJust mg f = maybe (return ()) f mg
|
|
|
|
-- | Conditionally run an action, using a X event to decide
|
|
whenX :: X Bool -> X () -> X ()
|
|
whenX a f = a >>= \b -> when b f
|
|
|
|
-- | 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
|
|
|
|
-- | A 'trace' for the X monad. Logs a string to stderr. The result may
|
|
-- be found in your .xsession-errors file
|
|
trace :: String -> X ()
|
|
trace msg = io $! do hPutStrLn stderr msg; hFlush stderr
|