aboutsummaryrefslogtreecommitdiffstats
path: root/XMonad
diff options
context:
space:
mode:
Diffstat (limited to 'XMonad')
-rw-r--r--XMonad/Config/PlainConfig.hs528
1 files changed, 0 insertions, 528 deletions
diff --git a/XMonad/Config/PlainConfig.hs b/XMonad/Config/PlainConfig.hs
deleted file mode 100644
index f1ebf1d..0000000
--- a/XMonad/Config/PlainConfig.hs
+++ /dev/null
@@ -1,528 +0,0 @@
-{-# LANGUAGE
- FlexibleInstances,
- FlexibleContexts,
- MultiParamTypeClasses,
- ExistentialQuantification
- #-}
-
--------------------------------------------------------------------------
--- |
--- Module : XMonad.Config.PlainConfig
--- Copyright : Braden Shepherdson <Braden.Shepherdson@gmail.com>
--- License : BSD3
---
--- Maintainer : Braden Shepherdson <Braden.Shepherdson@gmail.com>
---
--- Proof-of-concept (but usable) plain-text configuration file
--- parser, for use instead of xmonad.hs. Does not require recompilation,
--- allowing xmonad to be free of the GHC dependency.
---
--------------------------------------------------------------------------
-
-
-module XMonad.Config.PlainConfig
- (
- -- * Introduction
- -- $usage
-
- -- * Supported Layouts
- -- $layouts
-
- -- * Support Key Bindings
- -- $keys
-
- -- * Other Notes
- -- $notes
-
- -- * Example Config File
- -- $example
-
- plainConfig ,readConfig, checkConfig
- )
-where
-
-
-import XMonad
-import System.Exit
-
-import qualified XMonad.StackSet as W
-import qualified Data.Map as M
-import Data.List
-import Data.Maybe (isJust,fromJust)
-import Data.Char (isSpace)
-
-
---import Control.Monad
-import Control.Monad.Error
-import Control.Monad.Identity
-
-import Control.Arrow ((&&&))
-
-import Text.ParserCombinators.ReadP
-
-import System.IO
-import Control.Exception (bracket)
-
-import XMonad.Util.EZConfig (mkKeymap)
-
-
-
--- $usage
--- The @xmonad.hs@ file is very minimal when used with PlainConfig.
--- It typically contains only the following:
---
--- > module Main where
--- > import XMonad
--- > import XMonad.Config.PlainConfig (plainConfig)
--- > main = plainConfig
---
--- The 'plainConfig' function parses @~\/.xmonad\/xmonad.conf@,
--- the format of which is described below.
-
-
--- $layouts
--- Only 'Tall', 'Wide' and 'Full' are supported at present.
-
-
-
--- $keys
---
--- Key bindings are specified as a pair of an arbitrary EZConfig and
--- one of the following:
---
--- @ Name Haskell equivalent Default binding(s)@
---
--- * @spawn \<cmd\> spawn \"\<cmd\>\" none@
---
--- * @kill kill M-S-c@
---
--- * @nextLayout sendMessage NextLayout M-\<Space\>@
---
--- * @refresh refresh M-S-\<Space\>@
---
--- * @focusDown windows W.focusDown M-\<Tab\>, M-j@
---
--- * @focusUp windows W.focusUp M-k@
---
--- * @focusMaster windows W.focusMaster M-m@
---
--- * @swapDown windows W.swapDown M-S-j@
---
--- * @swapUp windows W.swapUp M-S-k@
---
--- * @swapMaster windows W.swapMaster M-\<Return\>@
---
--- * @shrink sendMessage Shrink M-h@
---
--- * @expand sendMessage Expand M-l@
---
--- * @sink withFocused $ windows . W.sink M-t@
---
--- * @incMaster sendMessage (IncMasterN 1) M-,@
---
--- * @decMaster sendMessage (IncMasterN (-1)) M-.@
---
--- * @quit io $ exitWith ExitSuccess M-S-q@
---
--- * @restart broadcastMessageReleaseResources >> restart \"xmonad\" True M-q@
---
-
-
--- $notes
--- Submaps are allowed.
--- These settings override the defaults. Changes made here will be used over
--- the default bindings for those keys.
-
-
--- $example
--- An example @~\/.xmonad\/xmonad.conf@ file follows:
---
--- @modMask = 3@
---
--- @numlockMask = 2@
---
--- @borderWidth = 1@
---
--- @normalBorderColor = #dddddd@
---
--- @focusedBorderColor = #00ff00@
---
--- @terminal=urxvt@
---
--- @workspaces=[\"1: IRC\",\"2: Web\", \"3\", \"4\", \"5\", \"6\", \"7\", \"8\", \"9\"]@
---
--- @focusFollowsMouse=True@
---
--- @layouts=[\"Tall\",\"Full\",\"Wide\"]@
---
--- @key=(\"M-x t\", \"spawn xmessage Test\")@
---
--- @manageHook=(ClassName \"MPlayer\" , \"float\" )@
---
--- @manageHook=(ClassName \"Gimp\" , \"float\" )@
---
--- @manageHook=(Resource \"desktop_window\", \"ignore\" )@
---
--- @manageHook=(Resource \"kdesktop\" , \"ignore\" )@
---
--- @manageHook=(Resource \"gnome-panel\" , \"ignore\" )@
---
-
-
-
-
-
-
-----------------------------------------------------------------
------- Several functions for parsing the key-value file. -------
-----------------------------------------------------------------
-
-parseKVBy :: Char -> ReadP (String,String)
-parseKVBy sep = do
- skipSpaces
- k <- munch1 (\x -> x /= ' ' && x /= sep)
- skipSpaces
- char kvSep
- skipSpaces
- v <- munch1 (\x -> x /= ' ') --or EOS
- return (k,v)
-
-parseKVVBy :: Char -> ReadP (String,String)
-parseKVVBy sep = do
- skipSpaces
- k <- munch1 (\x -> x /= ' ' && x /= sep)
- skipSpaces
- char kvSep
- skipSpaces
- v <- munch1 (const True) -- until EOS
- return (k,v)
-
-
-kvSep :: Char
-kvSep = '='
-
-parseKV, parseKVV :: ReadP (String,String)
-parseKV = parseKVBy kvSep
-parseKVV = parseKVVBy kvSep
-
-
-
-readKV :: String -> Integer -> RC (String,String)
-readKV s ln = case readP_to_S parseKV s of
- [((k,v),"")] -> return (k,v) --single, correct parse
- [] -> throwError [(ln,"No parse")]
- _ -> do
- case readP_to_S parseKVV s of
- [((k,v),"")] -> return (k,v) --single, correct parse
- [] -> throwError [(ln,"No parse")]
- xs -> throwError [(ln,"Ambiguous parse: "
- ++ show xs)]
-
-
-
-isComment :: String -> Bool
-isComment = not . null . readP_to_S parseComment
- where parseComment = skipSpaces >> char '#' >> return ()
- -- null means failed parse, so _not_ a comment.
-
-
-isBlank :: String -> Bool
-isBlank = null . filter (not . isSpace)
-
-
-type RC = ErrorT [(Integer,String)] Identity
-
-instance Error [(Integer,String)] where
- noMsg = [(-1, "Unknown error.")]
- strMsg s = [(-1, s)]
-
-
-parseFile :: [String] -> RC (XConfig Layout)
-parseFile ss = parseLines baseConfig theLines
- where theLines = filter (not . liftM2 (||) isComment isBlank . snd)
- $ zip [1..] ss
-
-
-
-parseLines :: XConfig Layout -> [(Integer,String)] -> RC (XConfig Layout)
-parseLines = foldM parse
-
-
-parse :: XConfig Layout -> (Integer, String) -> RC (XConfig Layout)
-parse xc (ln,s) = do
- (k,v) <- readKV s ln
- case M.lookup k commands of
- Nothing -> throwError [(ln,"Unknown command: "++k)]
- Just f -> f v ln xc
-
-
-
-
-----------------------------------------------------------------
--- Now the semantic parts, that convert from the relevant --
--- key-value entries to values in an XConfig --
-----------------------------------------------------------------
-
-
-
-type Command = String -> Integer -> XConfig Layout -> RC (XConfig Layout)
-
-commands :: M.Map String Command
-commands = M.fromList $
- [("modMask" , cmd_modMask )
- ,("numlockMask" , cmd_numlockMask )
- ,("normalBorderColor" , cmd_normalBorderColor )
- ,("focusedBorderColor" , cmd_focusedBorderColor)
- ,("terminal" , cmd_terminal )
- ,("workspaces" , cmd_workspaces )
- ,("focusFollowsMouse" , cmd_focusFollowsMouse )
- ,("layouts" , cmd_layouts )
- ,("key" , cmd_key )
- ,("manageHook" , cmd_manageHook )
- ,("borderWidth" , cmd_borderWidth )
- ]
-
-
--- | Behind-the-scenes helper for both 'cmd_modMask' and 'cmd_numlockMask'.
-genericModKey :: (KeyMask -> XConfig Layout) -> Command
-genericModKey f s ln _ = do
- x <- rcRead s ln :: RC Integer
- case lookup x (zip [1..] [mod1Mask,mod2Mask,mod3Mask,mod4Mask,mod5Mask]) of
- Just y -> return $ f y
- Nothing -> throwError [(ln,"Invalid mod key number: "++ show x)]
-
-
--- | Reads the mod key modifier number.
-cmd_modMask :: Command
-cmd_modMask s ln xc = genericModKey (\k -> xc{modMask = k}) s ln xc
-
--- | Reads the numlock key modifier number.
-cmd_numlockMask :: Command
-cmd_numlockMask s ln xc = genericModKey (\k -> xc{numlockMask = k}) s ln xc
-
-
--- | Reads the border width.
-cmd_borderWidth :: Command
-cmd_borderWidth s ln xc = do
- w <- rcRead s ln
- return $ xc { borderWidth = w }
-
-
--- | Reads the colors but just keeps them as RRGGBB Strings.
-cmd_normalBorderColor, cmd_focusedBorderColor :: Command
-cmd_normalBorderColor s _ xc = return $ xc{ normalBorderColor = s }
-cmd_focusedBorderColor s _ xc = return $ xc{ focusedBorderColor = s }
-
-
--- | Reads the terminal. It is just a String, no parsing.
-cmd_terminal :: Command
-cmd_terminal s _ xc = return $ xc{ terminal = s }
-
-
--- | Reads the workspace tag list. This is given as a Haskell [String].
-cmd_workspaces :: Command
-cmd_workspaces s ln xc = rcRead s ln >>= \x -> return xc{ workspaces = x }
-
-
--- | Reads the focusFollowsMouse, as a Haskell Bool.
-cmd_focusFollowsMouse :: Command
-cmd_focusFollowsMouse s ln xc = rcRead s ln >>=
- \x -> return xc{focusFollowsMouse = x}
-
-
--- | The list known layouts, mapped by name.
--- An easy location for improvement is to add more contrib layouts here.
-layouts :: M.Map String (Layout Window)
-layouts = M.fromList
- [("Tall", Layout (Tall 1 (3/100) (1/2)))
- ,("Wide", Layout (Mirror (Tall 1 (3/100) (1/2))))
- ,("Full", Layout Full)
- ]
-
-
--- | Expects a [String], the strings being layout names. Quotes required.
--- Draws from the `layouts' list above.
-cmd_layouts :: Command
-cmd_layouts s ln xc = do
- xs <- rcRead s ln -- read the list of strings
- let ls = map (id &&& (flip M.lookup) layouts) xs
- when (null ls) $ throwError [(ln,"Empty layout list")]
- case filter (not . isJust . snd) ls of
- [] -> return $ xc{ layoutHook = foldr1
- (\(Layout l) (Layout r) ->
- Layout (l ||| r)) (map (fromJust . snd) ls)
- }
- ys -> throwError $ map (\(x,_) -> (ln, "Unknown layout: "++ x)) ys
-
-
-
--- | A Map from names to key binding actions.
-key_actions :: M.Map String (X ())
-key_actions = M.fromList
- [("kill" , kill )
- ,("nextLayout" , sendMessage NextLayout )
- --,("prevLayout" , sendMessage PrevLayout )
- --,("resetLayout" , setLayout $ XMonad.layoutHook conf)
- ,("refresh" , refresh )
- ,("focusDown" , windows W.focusDown )
- ,("focusUp" , windows W.focusUp )
- ,("focusMaster" , windows W.focusMaster )
- ,("swapMaster" , windows W.swapMaster )
- ,("swapDown" , windows W.swapDown )
- ,("swapUp" , windows W.swapUp )
- ,("shrink" , sendMessage Shrink )
- ,("expand" , sendMessage Expand )
- ,("sink" , withFocused $ windows . W.sink)
- ,("incMaster" , sendMessage (IncMasterN 1))
- ,("decMaster" , sendMessage (IncMasterN (-1)))
- ,("quit" , io $ exitWith ExitSuccess)
- ,("restart" , broadcastMessage ReleaseResources
- >> restart "xmonad" True)
- ]
-
-
--- | Expects keys as described in the preamble, as
--- (\"EZConfig key name\", \"action name\"),
--- eg. (\"M-S-t\", \"spawn thunderbird\")
--- One key per "key=" line.
-cmd_key :: Command
-cmd_key s ln xc = do
- (k,v) <- rcRead s ln
- if "spawn " `isPrefixOf` v
- then return $ xc {
- keys = \c -> M.union (mkKeymap c
- [(k, spawn (drop 6 v))]
- ) ((keys xc) c)
- }
- else do
- case M.lookup v key_actions of
- Nothing -> throwError [(ln, "Unknown key action \"" ++ v ++ "\"")]
- Just ac -> return $
- xc { keys = \c -> M.union (mkKeymap c [(k, ac)])
- ((keys xc) c)
- }
-
-
-
--- | Map of names to actions for 'ManageHook's.
-manageHook_actions :: M.Map String ManageHook
-manageHook_actions = M.fromList
- [("float" , doFloat )
- ,("ignore" , doIgnore )
- ]
-
-
--- | Parses 'ManageHook's in the form given in the preamble.
--- eg. (ClassName \"MPlayer\", \"float\")
-cmd_manageHook :: Command
-cmd_manageHook s ln xc = do
- (k,v) <- rcRead s ln
- let q = parseQuery k
- if "toWorkspace " `isPrefixOf` v
- then return $ xc { manageHook = manageHook xc <+>
- (q --> doShift (drop 12 v))
- }
- else case M.lookup v manageHook_actions of
- Nothing -> throwError [(ln, "Unknown ManageHook action \""
- ++ v ++ "\"")]
- Just ac -> return $ xc { manageHook = manageHook xc <+> (q --> ac) }
-
-
-
--- | Core of the ManageHook expression parser.
--- Taken from Roman Cheplyaka's WindowProperties
-parseQuery :: Property -> Query Bool
-parseQuery (Title s) = title =? s
-parseQuery (ClassName s) = className =? s
-parseQuery (Resource s) = resource =? s
-parseQuery (And p q) = parseQuery p <&&> parseQuery q
-parseQuery (Or p q) = parseQuery p <&&> parseQuery q
-parseQuery (Not p) = not `fmap` parseQuery p
-parseQuery (Const b) = return b
-
-
--- | Property constructors are quite self-explaining.
--- Taken from Roman Cheplyaka's WindowProperties
-data Property = Title String
- | ClassName String
- | Resource String
- | And Property Property
- | Or Property Property
- | Not Property
- | Const Bool
- deriving (Read, Show)
-
-
-
--- | A wrapping of the read function into the RC monad.
-rcRead :: (Read a) => String -> Integer -> RC a
-rcRead s ln = case reads s of
- [(x,"")] -> return x
- _ -> throwError [(ln, "Failed to parse value")]
-
-
-
--- | The standard Config.hs 'defaultConfig', with the layout wrapped.
-baseConfig :: XConfig Layout
-baseConfig = defaultConfig{ layoutHook = Layout (layoutHook defaultConfig) }
-
-
-
--- | Core function that attempts to parse @~\/.xmonad\/xmonad.conf@
-readConfig :: IO (Maybe (XConfig Layout))
-readConfig = do
- dir <- getXMonadDir
- cs <- bracket (openFile (dir++"/xmonad.conf") ReadMode)
- (\h -> hClose h) -- vv force the lazy IO
- (\h -> (lines `fmap` hGetContents h) >>= \ss ->
- length ss `seq` return ss)
- let xce = runIdentity $ runErrorT $ parseFile cs
- case xce of
- Left es -> mapM_ (\(ln,e) ->
- putStrLn $ "readConfig error: line "++show ln++
- ": "++ e) es
- >> return Nothing
- Right xc -> return $ Just xc
-
-
--- | Attempts to run readConfig, and checks if it failed.
-checkConfig :: IO Bool
-checkConfig = isJust `fmap` readConfig
-
-
-
-{- REMOVED: It was for debugging, and causes an 'orphaned instances'
- warning to boot.
-
-
-
--- | Reads in the config, and then prints the resulting XConfig
-dumpConfig :: IO ()
-dumpConfig = readConfig >>= print
-
-
-instance Show (XConfig Layout) where
- show x = "XConfig { "
- ++ "normalBorderColor = "++ normalBorderColor x ++", "
- ++ "focusedBorderColor = "++ focusedBorderColor x++", "
- ++ "terminal = "++ terminal x ++", "
- ++ "workspaces = "++ show (workspaces x) ++", "
- ++ "numlockMask = "++ show (numlockMask x) ++", "
- ++ "modMask = "++ show (modMask x) ++", "
- ++ "borderWidth = "++ show (borderWidth x) ++", "
- ++ "focusFollowsMouse = "++ show (focusFollowsMouse x) ++", "
- ++ "layouts = "++ show (layoutHook x) ++" }"
-
--}
-
--- | Handles the unwrapping of the Layout. Intended for use as
--- @main = plainConfig@
-plainConfig :: IO ()
-plainConfig = do
- conf <- readConfig
- case conf of
- (Just xc@XConfig{layoutHook= (Layout l)}) ->
- xmonad (xc{ layoutHook = l })
- Nothing ->
- spawn $ "xmessage Failed to read xmonad.conf. See xmonad.errors."
-