aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--XMonad/Util/EZConfig.hs16
-rw-r--r--XMonad/Util/NamedActions.hs92
2 files changed, 57 insertions, 51 deletions
diff --git a/XMonad/Util/EZConfig.hs b/XMonad/Util/EZConfig.hs
index cf7545c..6e0a4c3 100644
--- a/XMonad/Util/EZConfig.hs
+++ b/XMonad/Util/EZConfig.hs
@@ -359,23 +359,19 @@ mkNamedKeymap c = mkNamedSubmaps . readKeymap c
-- group them into submaps in the appropriate way.
mkNamedSubmaps :: [([(KeyMask, KeySym)], NamedAction)] -> [((KeyMask, KeySym), NamedAction)]
-mkNamedSubmaps binds = map combine gathered
- where gathered = groupBy fstKey
- . sortBy (comparing fst)
- $ binds
- combine [([k],act)] = (k,act)
- combine ks = (head . fst . head $ ks,
- submapName . mkNamedSubmaps $ map (first tail) ks)
- fstKey = (==) `on` (head . fst)
+mkNamedSubmaps = mkSubmaps' submapName
mkSubmaps :: [ ([(KeyMask,KeySym)], X ()) ] -> [((KeyMask, KeySym), X ())]
-mkSubmaps binds = map combine gathered
+mkSubmaps = mkSubmaps' $ submap . M.fromList
+
+mkSubmaps' :: (Ord a) => ([(a, c)] -> c) -> [([a], c)] -> [(a, c)]
+mkSubmaps' subm binds = map combine gathered
where gathered = groupBy fstKey
. sortBy (comparing fst)
$ binds
combine [([k],act)] = (k,act)
combine ks = (head . fst . head $ ks,
- submap . M.fromList . mkSubmaps $ map (first tail) ks)
+ subm . mkSubmaps' subm $ map (first tail) ks)
fstKey = (==) `on` (head . fst)
on :: (a -> a -> b) -> (c -> a) -> c -> c -> b
diff --git a/XMonad/Util/NamedActions.hs b/XMonad/Util/NamedActions.hs
index 089d7c8..70544e4 100644
--- a/XMonad/Util/NamedActions.hs
+++ b/XMonad/Util/NamedActions.hs
@@ -1,17 +1,17 @@
{-# OPTIONS_GHC -fno-warn-orphans #-}
-{-# LANGUAGE FlexibleInstances, StandaloneDeriving #-}
-{-# LANGUAGE ExistentialQuantification, FlexibleContexts #-}
+{-# LANGUAGE ExistentialQuantification, FlexibleContexts, FlexibleInstances, StandaloneDeriving #-}
--------------------------------------------------------------------
-- |
-- Module : XMonad.Util.NamedActions
--- Copyright : Adam Vogt <vogt.adam@gmail.com>
+-- Copyright : 2009 Adam Vogt <vogt.adam@gmail.com>
-- License : BSD3-style (see LICENSE)
--
-- Maintainer : Adam Vogt <vogt.adam@gmail.com>
-- Stability : unstable
-- Portability : unportable
--
--- Present a list of the keybindings in use.
+-- A wrapper for keybinding configuration that can list the available
+-- keybindings.
--------------------------------------------------------------------
module XMonad.Util.NamedActions (
@@ -40,9 +40,10 @@ module XMonad.Util.NamedActions (
defaultKeysDescr
) where
+
import XMonad.Actions.Submap(submap)
import XMonad(KeySym, KeyMask, X, Layout, Message,
- XConfig(workspaces, terminal, modMask, layoutHook, keys, XConfig),
+ XConfig(keys, layoutHook, modMask, terminal, workspaces, XConfig),
io, spawn, whenJust, ChangeLayout(NextLayout), IncMasterN(..),
Resize(..), kill, refresh, screenWorkspace, sendMessage, setLayout,
windows, withFocused, controlMask, mod1Mask, mod2Mask, mod3Mask,
@@ -50,15 +51,18 @@ import XMonad(KeySym, KeyMask, X, Layout, Message,
xK_comma, xK_e, xK_h, xK_j, xK_k, xK_l, xK_m, xK_n, xK_p,
xK_period, xK_q, xK_r, xK_space, xK_t, xK_w, keysymToString)
import System.Posix.Process(executeFile, forkProcess)
-import Control.Arrow(Arrow((***), second, (&&&), first))
-import Data.Bits(Bits((.|.), complement, (.&.)))
-import Data.Function((.), const, ($), flip, id, on)
-import Data.List((++), filter, zip, map, concatMap, elem, head,
- last, null, unlines, groupBy, intercalate, partition, sortBy)
+import Control.Arrow(Arrow((&&&), second, (***)))
+import Data.Bits(Bits((.&.), complement, (.|.)))
+import Data.Function((.), const, ($), flip, id)
+import Data.List((++), filter, zip, map, concatMap, null, unlines,
+ groupBy)
import System.Exit(ExitCode(ExitSuccess), exitWith)
+import Control.Applicative ((<*>))
+
import qualified Data.Map as M
import qualified XMonad.StackSet as W
+import qualified XMonad
-- $usage
-- Here is an example config that demonstrates the usage of 'sendMessage'',
@@ -68,34 +72,35 @@ import qualified XMonad.StackSet as W
-- > import XMonad.Util.NamedActions
-- > import XMonad.Util.EZConfig
-- >
--- > main = xmonad $ addDescrKeys ((mod4Mask, xK_d), xMessage) myKeys
+-- > main = xmonad $ addDescrKeys ((mod4Mask, xK_F1), xMessage) myKeys
-- > defaultConfig { modMask = mod4Mask }
-- >
--- > myKeys = flip mkNamedKeymap $
--- > [("M-x a", addName "useless..." $ spawn "xmessage foo"),
+-- > myKeys c = (subtitle "Custom Keys":) $ mkNamedKeymap c $
+-- > [("M-x a", addName "useless message" $ spawn "xmessage foo"),
-- > ("M-c", sendMessage' Expand)]
-- > ^++^
-- > [("<XF86AudioPlay>", spawn "mpc toggle" :: X ()),
--- > ("<XF86AudioNext>", spawn "mpc next"]
+-- > ("<XF86AudioNext>", spawn "mpc next")]
--
--- Due to the type of '^++^', you can combine bindings whose actions are @X ()@
+-- Using '^++^', you can combine bindings whose actions are @X ()@
-- as well as actions that have descriptions. However you cannot mix the two in
--- a single list, unless each is prefixed with 'addName' or 'noName'. '^++^'
--- works with traditional-style keybindings too.
+-- a single list, unless each is prefixed with 'addName' or 'noName'.
+--
+-- If you don't like EZConfig, you can still use '^++^' with the basic XMonad
+-- keybinding configuration too.
--
-- Also note the unfortunate necessity of a type annotation, since 'spawn' is
-- too general.
-- TODO: squeeze titles that have no entries (consider titles containing \n)
--
--- pad as if by columns
---
--- Multiple columns
+-- Output to Multiple columns
--
-- Devin Mullin's suggestions:
--
---Reduce redundancy wrt mkNamedSubmaps, mkSubmaps and mkNamedKeymap to have a
---HasName context (and leave mkKeymap as a specific case of it?)
+-- Reduce redundancy wrt mkNamedSubmaps, mkSubmaps and mkNamedKeymap to have a
+-- HasName context (and leave mkKeymap as a specific case of it?)
+-- Currently kept separate to aid error messages, common lines factored out
--
-- Suggestions for UI:
--
@@ -171,7 +176,7 @@ a ^++^ b = map (second NamedAction) a ++ map (second NamedAction) b
-- | Or allow another lookup table?
modToString :: KeyMask -> String
modToString mask = concatMap (++"-") $ filter (not . null)
- $ map (uncurry w)
+ $ map (uncurry pick)
[(mod1Mask, "M1")
,(mod2Mask, "M2")
,(mod3Mask, "M3")
@@ -179,26 +184,30 @@ modToString mask = concatMap (++"-") $ filter (not . null)
,(mod5Mask, "M5")
,(controlMask, "C")
,(shiftMask,"Shift")]
- where w m str = if m .&. complement mask == 0 then str else ""
+ where pick m str = if m .&. complement mask == 0 then str else ""
keyToString :: (KeyMask, KeySym) -> [Char]
keyToString = uncurry (++) . (modToString *** keysymToString)
--- | Squeezes bindings from [xK_1 .. xK_9]
-showKm :: [((KeyMask, KeySym), NamedAction)] -> [[Char]]
-showKm = uncurry (flip (++))
- . second showKmSimple
- . first (map ( intercalate " ... " . showKmSimple . uncurry (:)
- . (head &&& (:[]) . last)
- . sortBy (compare `on` (snd . fst)))
- . groupBy ((==) `on` (fst . fst))
- )
- . partition ((`elem` [xK_1 .. xK_9]) . snd . fst)
-
showKmSimple :: [((KeyMask, KeySym), NamedAction)] -> [[Char]]
showKmSimple = concatMap (\(k,e) -> if snd k == 0 then "":showName e else map ((keyToString k ++) . smartSpace) $ showName e)
- where smartSpace [] = []
- smartSpace xs = ' ':xs
+
+smartSpace :: String -> String
+smartSpace [] = []
+smartSpace xs = ' ':xs
+
+_test :: String
+_test = unlines $ showKm $ defaultKeysDescr XMonad.defaultConfig { XMonad.layoutHook = XMonad.Layout $ XMonad.layoutHook XMonad.defaultConfig }
+
+showKm :: [((KeyMask, KeySym), NamedAction)] -> [String]
+showKm keybindings = padding $ do
+ (k,e) <- keybindings
+ if snd k == 0 then map ((,) "") $ showName e
+ else map ((,) (keyToString k) . smartSpace) $ showName e
+ where padding = let pad n (k,e) = if null k then "\n>> "++e else take n (k++repeat ' ') ++ e
+ expand xs n = map (pad n) xs
+ getMax = map (maximum . map (length . fst))
+ in concat . (zipWith expand <*> getMax) . groupBy (const $ not . null . fst)
-- | An action to send to 'addDescrKeys' for showing the keybindings. See also 'showKm' and 'showKmSimple'
xMessage :: [((KeyMask, KeySym), NamedAction)] -> NamedAction
@@ -206,7 +215,8 @@ xMessage x = addName "Show Keybindings" $ io $ do
forkProcess $ executeFile "xmessage" True ["-default", "okay", unlines $ showKm x] Nothing
return ()
--- | Merge the supplied keys with 'defaultKeysDescr'
+-- | Merge the supplied keys with 'defaultKeysDescr', also adding a keybinding
+-- to run an action for showing the keybindings.
addDescrKeys :: (HasName b1, HasName b) =>
((KeyMask, KeySym),[((KeyMask, KeySym), NamedAction)] -> b)
-> (XConfig Layout -> [((KeyMask, KeySym), b1)])
@@ -259,7 +269,7 @@ defaultKeysDescr conf@(XConfig {XMonad.modMask = modm}) =
, subtitle "floating layer support"
, ((modm, xK_t ), addName "Push floating to tiled" $ withFocused $ windows . W.sink) -- %! Push window back into tiling
- , subtitle "increase or decrease number of windows in the master area"
+ , subtitle "change the number of windows in the master area"
, ((modm , xK_comma ), sendMessage' (IncMasterN 1)) -- %! Increment the number of windows in the master area
, ((modm , xK_period), sendMessage' (IncMasterN (-1))) -- %! Deincrement the number of windows in the master area
@@ -284,8 +294,8 @@ defaultKeysDescr conf@(XConfig {XMonad.modMask = modm}) =
, (key, sc) <- zip [xK_w, xK_e, xK_r] [0..]]
-- | For a prettier presentation: keymask, keysym of 0 are reserved for this
--- purpose: they do not happen, afaik, and keysymToString 0 would raise error
--- otherwise
+-- purpose: they do not happen, afaik, and keysymToString 0 would raise an
+-- error otherwise
separator :: ((KeyMask,KeySym), NamedAction)
separator = ((0,0), NamedAction (return () :: X (),[] :: [String]))