mirror of
https://github.com/xmonad/xmonad.git
synced 2025-08-03 13:41:54 -07:00
Compare commits
345 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
8399e80327 | ||
|
3e3d516092 | ||
|
d2ae7310d6 | ||
|
ca3e277d2b | ||
|
bb2b6c7bf8 | ||
|
d74814af35 | ||
|
f9799422f9 | ||
|
be5e27038f | ||
|
1f4b8cb5f6 | ||
|
da7ca1c29d | ||
|
e095621ab9 | ||
|
93c55c948e | ||
|
9ff105340e | ||
|
5e61b137fb | ||
|
aeef36f74c | ||
|
673f303646 | ||
|
7f3c6823d4 | ||
|
79f23d6cec | ||
|
46f5e68cfa | ||
|
76d2bddaf0 | ||
|
f5e55f3a27 | ||
|
6c72a03fb1 | ||
|
31c7734f7b | ||
|
d1af7d986d | ||
|
da167bfc11 | ||
|
c46f3ad549 | ||
|
5b42a58d06 | ||
|
e8292e0e9d | ||
|
6cd46e12bb | ||
|
2441275122 | ||
|
f70ab7964e | ||
|
237fdbf037 | ||
|
5166ede96b | ||
|
56463b2391 | ||
|
f427c2b0e9 | ||
|
287d364e0d | ||
|
8c31768b79 | ||
|
9ceef229c3 | ||
|
40581c9bf8 | ||
|
161ade3593 | ||
|
f2461c9e3a | ||
|
11b37429b1 | ||
|
bbf5d0010c | ||
|
2f60ee5680 | ||
|
3e2d48d5da | ||
|
462422d07a | ||
|
33f28ed2ac | ||
|
a29590034a | ||
|
f394956e56 | ||
|
039d9e2b96 | ||
|
a73f8ec709 | ||
|
1bb18654d6 | ||
|
fa45d59e95 | ||
|
f73f8f38a5 | ||
|
28cc666a75 | ||
|
c8f16a85cf | ||
|
6908189698 | ||
|
39eccc350c | ||
|
c8ab301c95 | ||
|
5e310c0c94 | ||
|
4fa10442ab | ||
|
1ab1d729a0 | ||
|
c95b8d9160 | ||
|
92b4510d7b | ||
|
6114bb371e | ||
|
7e2ec3840c | ||
|
6ce125a566 | ||
|
3456086f85 | ||
|
3b83895d28 | ||
|
dc6ba6b5ee | ||
|
df5003eb16 | ||
|
99dd1a30ba | ||
|
d6c5eb3e80 | ||
|
9d9b733994 | ||
|
ea71fd67e8 | ||
|
e9eadd6141 | ||
|
ddf9e49e49 | ||
|
81803ffe81 | ||
|
31ce83d04e | ||
|
c2ae7a8c71 | ||
|
45eea722be | ||
|
4bb6371155 | ||
|
dfd4d435d8 | ||
|
ac41c8fb52 | ||
|
223b48ab27 | ||
|
107b942414 | ||
|
6aee5509de | ||
|
ba6d9c8a52 | ||
|
3a995b40c9 | ||
|
656f4551da | ||
|
6ae94edbe4 | ||
|
22ccca29e6 | ||
|
2302bb3304 | ||
|
b4e0e77911 | ||
|
dcf53fbaf6 | ||
|
833e37da9c | ||
|
cf0c3b9ab6 | ||
|
532a920bce | ||
|
0d506daf45 | ||
|
4887c5ac42 | ||
|
3d0c08365d | ||
|
e4c2a81ca1 | ||
|
58fc2bc59e | ||
|
c8473e3ae9 | ||
|
11711e1a46 | ||
|
99fb75eb9b | ||
|
ceb1c51b3f | ||
|
14b6306ac2 | ||
|
b51f6f55a8 | ||
|
e2ab6e8a27 | ||
|
a5200b3862 | ||
|
f81ec95fa0 | ||
|
39f4fe7a90 | ||
|
d50d6c909d | ||
|
dbfd13207d | ||
|
6eb23670bb | ||
|
bbe4a27f65 | ||
|
94924123bb | ||
|
ece268cd1e | ||
|
dfd8e51136 | ||
|
0de10862c2 | ||
|
f7b6a4508f | ||
|
00f83ac78a | ||
|
3a902ce613 | ||
|
5342be0e67 | ||
|
88845e5d97 | ||
|
4732557c12 | ||
|
a13c11ff52 | ||
|
fcea17f920 | ||
|
a5acef3ad6 | ||
|
76e960a40c | ||
|
30af3a8f84 | ||
|
c9142952c2 | ||
|
934fb2c368 | ||
|
d1c29a40cf | ||
|
cd9c592ebc | ||
|
131e060533 | ||
|
4996b1bc47 | ||
|
4b2366b5ce | ||
|
0590f5da9e | ||
|
c3c39aae12 | ||
|
7bc4ab41c7 | ||
|
7dc2d254d1 | ||
|
528d51e58a | ||
|
9ef3fdcf08 | ||
|
e8d3f674ef | ||
|
8a5d2490bb | ||
|
22aacf9bf6 | ||
|
b0b43050f4 | ||
|
23035e944b | ||
|
bf52d34bbf | ||
|
8a8c538c23 | ||
|
e50927ffc0 | ||
|
3789f37f25 | ||
|
48ccbc7fb2 | ||
|
d679ceb234 | ||
|
7b3c1243b7 | ||
|
97fe14dfd2 | ||
|
c1e039ba88 | ||
|
9bd11aeea5 | ||
|
c350caf9b8 | ||
|
066da1cd99 | ||
|
cadf81976f | ||
|
ddd1fa9cae | ||
|
3bd63adb60 | ||
|
e384a358b5 | ||
|
8971ab7fae | ||
|
2e8794d0f3 | ||
|
92d58ae0a8 | ||
|
33e14e7ba7 | ||
|
ec45881d4c | ||
|
b73ac809ba | ||
|
d0507c9eb3 | ||
|
fc82a7d412 | ||
|
cc019f487c | ||
|
4c7cf15cdb | ||
|
350a4d6f6b | ||
|
1ddaffbfba | ||
|
0903c76d40 | ||
|
156a89b761 | ||
|
1ea1c05617 | ||
|
f5ad470815 | ||
|
1be4bc5d91 | ||
|
0514380d76 | ||
|
18cf8fbb10 | ||
|
eb65473591 | ||
|
c734586275 | ||
|
74131eb15f | ||
|
ac94932345 | ||
|
bec871d254 | ||
|
258f85dd08 | ||
|
16abab4241 | ||
|
01cf4a5581 | ||
|
bd6a52e587 | ||
|
0938298f29 | ||
|
5bd96a8e1a | ||
|
874e6f80f0 | ||
|
6fecf7c425 | ||
|
3a18204adb | ||
|
2599706141 | ||
|
2edc5a92c2 | ||
|
b6d36f3c70 | ||
|
5c9850bf6d | ||
|
3f3b4251c2 | ||
|
2de6cc7cf1 | ||
|
3aa746c0db | ||
|
48b001f9a2 | ||
|
775172983b | ||
|
f5bec53b83 | ||
|
77e3876d07 | ||
|
f439e766a4 | ||
|
c436e63a15 | ||
|
d610407cf8 | ||
|
f7d6f6b6f7 | ||
|
6dba9ddeb3 | ||
|
49f64197b2 | ||
|
1f625a6c0d | ||
|
1eaee82e85 | ||
|
07be5998c0 | ||
|
6d7307030a | ||
|
6c94b3b217 | ||
|
1a48b527ff | ||
|
75874040cc | ||
|
e331dd4a82 | ||
|
3cf5c1f9d4 | ||
|
4cfe583f63 | ||
|
d348f2ae72 | ||
|
41063f2e57 | ||
|
42dde26d4d | ||
|
c66ff8335e | ||
|
f7ecf70a35 | ||
|
fd10c198e6 | ||
|
c49b8f567f | ||
|
1d0191184f | ||
|
6c38226553 | ||
|
5cd9094f58 | ||
|
053f1adb7c | ||
|
6294e6adf5 | ||
|
1f9e77bd90 | ||
|
8d3b6fa304 | ||
|
6316d4f2ff | ||
|
a88a0b1b8b | ||
|
bab04b71d3 | ||
|
d83ce46a1e | ||
|
aaaeae54c3 | ||
|
e0bcad162f | ||
|
b07e334405 | ||
|
c237441003 | ||
|
42b691d515 | ||
|
65f3f4db8a | ||
|
172e046e84 | ||
|
7dac92057d | ||
|
8dbf8896c9 | ||
|
f11ce95528 | ||
|
d57aab25ef | ||
|
89645e0999 | ||
|
ab0ebe1050 | ||
|
8e303a6bea | ||
|
e70fb29efc | ||
|
2afa8c3a7a | ||
|
ce0d5d376d | ||
|
4bcad8fe60 | ||
|
da3db68b59 | ||
|
045ed777a2 | ||
|
e8bbba9694 | ||
|
8b3dc01e53 | ||
|
8b8433a9e7 | ||
|
acbe7976d7 | ||
|
d05b01431d | ||
|
60a40be09e | ||
|
2b6a200ad3 | ||
|
2196ab7469 | ||
|
ff1918ad20 | ||
|
5e3317b28e | ||
|
306b3c11c3 | ||
|
5ef7c5f5d0 | ||
|
16d4ce5706 | ||
|
6640e434bf | ||
|
cee31df81d | ||
|
029dd68860 | ||
|
bd64d169fe | ||
|
d0d81db6de | ||
|
f1c1e982a2 | ||
|
60dda50181 | ||
|
3b64981c78 | ||
|
a7c4c38ba8 | ||
|
34bbbf59c4 | ||
|
4fd7353d8e | ||
|
7e8de677cb | ||
|
3a9dc57c69 | ||
|
5f9222efb4 | ||
|
d1fdd4a020 | ||
|
2ab2195782 | ||
|
71bce5e525 | ||
|
9c78ba538b | ||
|
68c72b34e1 | ||
|
2caf68ee69 | ||
|
f420ae881d | ||
|
e062265b38 | ||
|
9c35abaa46 | ||
|
ee39e7fdb8 | ||
|
e6fb743e5a | ||
|
0b11d6666d | ||
|
4a7ec374d0 | ||
|
1c603ebc4b | ||
|
3af0ccf73c | ||
|
fe397edf4a | ||
|
70282f23dc | ||
|
f3f12383f0 | ||
|
cb13207644 | ||
|
eb1e38405d | ||
|
d43384cfc7 | ||
|
197c834331 | ||
|
5f12ca0faa | ||
|
b4929576e7 | ||
|
0e5f8b03e8 | ||
|
3f03dcb5c1 | ||
|
bee79c83e6 | ||
|
29a5256c10 | ||
|
6cff2dddcf | ||
|
d1ad738f6b | ||
|
3b6bfbf54c | ||
|
f8c0ae5407 | ||
|
f1aa00f96f | ||
|
5e943d512c | ||
|
019315e70c | ||
|
bc525b79e3 | ||
|
f67ebbf495 | ||
|
3060c36d00 | ||
|
c6f346f887 | ||
|
e87a111a50 | ||
|
3b5ca225f6 | ||
|
46ef80ad06 | ||
|
b72c096bc6 | ||
|
0842194940 | ||
|
2b207a28ef | ||
|
54af88d5f6 | ||
|
b6f00e9aab | ||
|
874a4264c3 | ||
|
6898a0e583 | ||
|
f668b6238a | ||
|
bccf8dd5f8 | ||
|
41e3b073c8 | ||
|
82dd5b8119 | ||
|
1fb52ce2cc |
82
CONFIG
Normal file
82
CONFIG
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
== Configuring xmonad ==
|
||||||
|
|
||||||
|
xmonad is configured by creating and editing the file:
|
||||||
|
|
||||||
|
~/.xmonad/xmonad.hs
|
||||||
|
|
||||||
|
xmonad then uses settings from this file as arguments to the window manager,
|
||||||
|
on startup. For a complete example of possible settings, see the file:
|
||||||
|
|
||||||
|
man/xmonad.hs
|
||||||
|
|
||||||
|
Further examples are on the website, wiki and extension documentation.
|
||||||
|
|
||||||
|
http://haskell.org/haskellwiki/Xmonad
|
||||||
|
|
||||||
|
== A simple example ==
|
||||||
|
|
||||||
|
Here is a basic example, which overrides the default border width,
|
||||||
|
default terminal, and some colours. This text goes in the file
|
||||||
|
$HOME/.xmonad/xmonad.hs :
|
||||||
|
|
||||||
|
import XMonad
|
||||||
|
|
||||||
|
main = xmonad $ defaultConfig
|
||||||
|
{ borderWidth = 2
|
||||||
|
, terminal = "urxvt"
|
||||||
|
, normalBorderColor = "#cccccc"
|
||||||
|
, focusedBorderColor = "#cd8b00" }
|
||||||
|
|
||||||
|
You can find the defaults in the file:
|
||||||
|
|
||||||
|
XMonad/Config.hs
|
||||||
|
|
||||||
|
== Checking your xmonad.hs is correct ==
|
||||||
|
|
||||||
|
Place this text in ~/.xmonad/xmonad.hs, and then check that it is
|
||||||
|
syntactically and type correct by loading it in the Haskell
|
||||||
|
interpreter:
|
||||||
|
|
||||||
|
$ ghci ~/.xmonad/xmonad.hs
|
||||||
|
GHCi, version 6.8.1: http://www.haskell.org/ghc/ :? for help
|
||||||
|
Loading package base ... linking ... done.
|
||||||
|
Ok, modules loaded: Main.
|
||||||
|
|
||||||
|
Prelude Main> :t main
|
||||||
|
main :: IO ()
|
||||||
|
|
||||||
|
Ok, looks good.
|
||||||
|
|
||||||
|
== Loading your configuration ==
|
||||||
|
|
||||||
|
To have xmonad start using your settings, type 'mod-q'. xmonad will
|
||||||
|
then load this new file, and run it. If it is unable to, the defaults
|
||||||
|
are used.
|
||||||
|
|
||||||
|
To load succesfully, both 'xmonad' and 'ghc' must be in your $PATH
|
||||||
|
environment variable. If GHC isn't in your path, for some reason, you
|
||||||
|
can compile the xmonad.hs file yourself:
|
||||||
|
|
||||||
|
$ cd ~/.xmonad
|
||||||
|
$ ghc --make xmonad.hs
|
||||||
|
$ ls
|
||||||
|
xmonad xmonad.hi xmonad.hs xmonad.o
|
||||||
|
|
||||||
|
When you hit mod-q, this newly compiled xmonad will be used.
|
||||||
|
|
||||||
|
== Where are the defaults? ==
|
||||||
|
|
||||||
|
The default configuration values are defined in the source file:
|
||||||
|
|
||||||
|
XMonad/Config.hs
|
||||||
|
|
||||||
|
the XConfig data structure itself is defined in:
|
||||||
|
|
||||||
|
XMonad/Core.hs
|
||||||
|
|
||||||
|
== Extensions ==
|
||||||
|
|
||||||
|
Since the xmonad.hs file is just another Haskell module, you may import
|
||||||
|
and use any Haskell code or libraries you wish. For example, you can use
|
||||||
|
things from the xmonad-contrib library, or other code you write
|
||||||
|
yourself.
|
186
Config.hs
186
Config.hs
@@ -1,186 +0,0 @@
|
|||||||
-----------------------------------------------------------------------------
|
|
||||||
-- |
|
|
||||||
-- Module : Config.hs
|
|
||||||
-- Copyright : (c) Spencer Janssen 2007
|
|
||||||
-- License : BSD3-style (see LICENSE)
|
|
||||||
--
|
|
||||||
-- Maintainer : dons@cse.unsw.edu.au
|
|
||||||
-- Stability : stable
|
|
||||||
-- Portability : portable
|
|
||||||
--
|
|
||||||
--
|
|
||||||
-- This module specifies configurable defaults for xmonad. If you change
|
|
||||||
-- values here, be sure to recompile and restart (mod-q) xmonad,
|
|
||||||
-- for the changes to take effect.
|
|
||||||
--
|
|
||||||
------------------------------------------------------------------------
|
|
||||||
|
|
||||||
module Config where
|
|
||||||
|
|
||||||
--
|
|
||||||
-- Useful imports
|
|
||||||
--
|
|
||||||
import XMonad
|
|
||||||
import Operations
|
|
||||||
import qualified StackSet as W
|
|
||||||
import Data.Ratio
|
|
||||||
import Data.Bits ((.|.))
|
|
||||||
import qualified Data.Map as M
|
|
||||||
import System.Exit
|
|
||||||
import Graphics.X11.Xlib
|
|
||||||
|
|
||||||
--
|
|
||||||
-- The number of workspaces (virtual screens, or window groups)
|
|
||||||
--
|
|
||||||
workspaces :: [WorkspaceId]
|
|
||||||
workspaces = [0..8]
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- modMask lets you specify which modkey you want to use. The default is
|
|
||||||
-- mod1Mask ("left alt"). You may also consider using mod3Mask ("right
|
|
||||||
-- alt"), which does not conflict with emacs keybindings. The "windows
|
|
||||||
-- key" is usually mod4Mask.
|
|
||||||
--
|
|
||||||
modMask :: KeyMask
|
|
||||||
modMask = mod1Mask
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- Default offset of drawable screen boundaries from each physical screen.
|
|
||||||
-- Anything non-zero here will leave a gap of that many pixels on the
|
|
||||||
-- given edge, on the that screen. A useful gap at top of screen for a
|
|
||||||
-- menu bar (e.g. 15)
|
|
||||||
--
|
|
||||||
-- An example, to set a top gap on monitor 1, and a gap on the bottom of
|
|
||||||
-- monitor 2, you'd use a list of geometries like so:
|
|
||||||
--
|
|
||||||
-- > defaultGaps = [(18,0,0,0),(0,18,0,0)] -- 2 gaps on 2 monitors
|
|
||||||
--
|
|
||||||
-- Fields are: top, bottom, left, right.
|
|
||||||
--
|
|
||||||
defaultGaps :: [(Int,Int,Int,Int)]
|
|
||||||
defaultGaps = [(0,0,0,0)] -- 15 for default dzen
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- numlock handling:
|
|
||||||
--
|
|
||||||
-- The mask for the numlock key. You may need to change this on some systems.
|
|
||||||
--
|
|
||||||
-- You can find the numlock modifier by running "xmodmap" and looking for a
|
|
||||||
-- modifier with Num_Lock bound to it:
|
|
||||||
--
|
|
||||||
-- $ xmodmap | grep Num
|
|
||||||
-- mod2 Num_Lock (0x4d)
|
|
||||||
--
|
|
||||||
numlockMask :: KeyMask
|
|
||||||
numlockMask = mod2Mask
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- Border colors for unfocused and focused windows, respectively.
|
|
||||||
--
|
|
||||||
normalBorderColor, focusedBorderColor :: String
|
|
||||||
normalBorderColor = "#dddddd"
|
|
||||||
focusedBorderColor = "#ff0000"
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- Width of the window border in pixels
|
|
||||||
--
|
|
||||||
borderWidth :: Dimension
|
|
||||||
borderWidth = 1
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- The default set of tiling algorithms
|
|
||||||
--
|
|
||||||
defaultLayouts :: [Layout Window]
|
|
||||||
defaultLayouts = [ tiled , mirror tiled , full ]
|
|
||||||
where
|
|
||||||
-- default tiling algorithm partitions the screen into two panes
|
|
||||||
tiled = tall nmaster delta ratio
|
|
||||||
|
|
||||||
-- The default number of windows in the master pane
|
|
||||||
nmaster = 1
|
|
||||||
|
|
||||||
-- Default proportion of screen occupied by master pane
|
|
||||||
ratio = 1%2
|
|
||||||
|
|
||||||
-- Percent of screen to increment by when resizing panes
|
|
||||||
delta = 3%100
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- Perform an arbitrary action on each state change.
|
|
||||||
-- Examples include:
|
|
||||||
-- * do nothing
|
|
||||||
-- * log the state to stdout
|
|
||||||
--
|
|
||||||
logHook :: X ()
|
|
||||||
logHook = return ()
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- The key bindings list.
|
|
||||||
--
|
|
||||||
-- The unusual comment format is used to generate the documentation
|
|
||||||
-- automatically.
|
|
||||||
--
|
|
||||||
keys :: M.Map (KeyMask, KeySym) (X ())
|
|
||||||
keys = M.fromList $
|
|
||||||
-- launching and killing programs
|
|
||||||
[ ((modMask .|. shiftMask, xK_Return), spawn "xterm") -- @@ Launch an xterm
|
|
||||||
, ((modMask, xK_p ), spawn "exe=`dmenu_path | dmenu` && eval \"exec $exe\"") -- @@ Launch dmenu
|
|
||||||
, ((modMask .|. shiftMask, xK_p ), spawn "gmrun") -- @@ Launch gmrun
|
|
||||||
, ((modMask .|. shiftMask, xK_c ), kill) -- @@ Close the focused window
|
|
||||||
|
|
||||||
, ((modMask, xK_space ), switchLayout) -- @@ Rotate through the available layout algorithms
|
|
||||||
|
|
||||||
, ((modMask, xK_n ), refresh) -- @@ Resize viewed windows to the correct size
|
|
||||||
|
|
||||||
-- move focus up or down the window stack
|
|
||||||
, ((modMask, xK_Tab ), focusDown) -- @@ Move focus to the next window
|
|
||||||
, ((modMask, xK_j ), focusDown) -- @@ Move focus to the next window
|
|
||||||
, ((modMask, xK_k ), focusUp ) -- @@ Move focus to the previous window
|
|
||||||
|
|
||||||
-- modifying the window order
|
|
||||||
, ((modMask, xK_Return), swapMaster) -- @@ Swap the focused window and the master window
|
|
||||||
, ((modMask .|. shiftMask, xK_j ), swapDown ) -- @@ Swap the focused window with the next window
|
|
||||||
, ((modMask .|. shiftMask, xK_k ), swapUp ) -- @@ Swap the focused window with the previous window
|
|
||||||
|
|
||||||
-- resizing the master/slave ratio
|
|
||||||
, ((modMask, xK_h ), sendMessage Shrink) -- @@ Shrink the master area
|
|
||||||
, ((modMask, xK_l ), sendMessage Expand) -- @@ Expand the master area
|
|
||||||
|
|
||||||
, ((modMask, xK_t ), withFocused sink) -- @@ Push window back into tiling
|
|
||||||
|
|
||||||
-- increase or decrease number of windows in the master area
|
|
||||||
, ((modMask , xK_comma ), sendMessage (IncMasterN 1)) -- @@ Increment the number of windows in the master area
|
|
||||||
, ((modMask , xK_period), sendMessage (IncMasterN (-1))) -- @@ Deincrement the number of windows in the master area
|
|
||||||
|
|
||||||
-- toggle the status bar gap
|
|
||||||
, ((modMask , xK_b ), modifyGap (\i n -> let x = (defaultGaps ++ repeat (0,0,0,0)) !! i in if n == x then (0,0,0,0) else x)) -- @@ Toggle the status bar gap
|
|
||||||
|
|
||||||
-- quit, or restart
|
|
||||||
, ((modMask .|. shiftMask, xK_q ), io (exitWith ExitSuccess)) -- @@ Quit xmonad
|
|
||||||
, ((modMask , xK_q ), restart Nothing True) -- @@ Restart xmonad
|
|
||||||
|
|
||||||
] ++
|
|
||||||
-- mod-[1..9] @@ Switch to workspace N
|
|
||||||
-- mod-shift-[1..9] @@ Move client to workspace N
|
|
||||||
[((m .|. modMask, k), f i)
|
|
||||||
| (i, k) <- zip workspaces [xK_1 ..]
|
|
||||||
, (f, m) <- [(view, 0), (shift, shiftMask)]]
|
|
||||||
|
|
||||||
-- mod-{w,e,r} @@ Switch to physical/Xinerama screens 1, 2, or 3
|
|
||||||
-- mod-shift-{w,e,r} @@ Move client to screen 1, 2, or 3
|
|
||||||
++
|
|
||||||
[((m .|. modMask, key), screenWorkspace sc >>= flip whenJust f)
|
|
||||||
| (key, sc) <- zip [xK_w, xK_e, xK_r] [0..]
|
|
||||||
, (f, m) <- [(windows . W.view, 0), (shift, shiftMask)]]
|
|
||||||
|
|
||||||
-- |
|
|
||||||
-- default actions bound to mouse events
|
|
||||||
--
|
|
||||||
mouseBindings :: M.Map (KeyMask, Button) (Window -> X ())
|
|
||||||
mouseBindings = M.fromList $
|
|
||||||
-- mod-button1 @@ Set the window to floating mode and move by dragging
|
|
||||||
[ ((modMask, button1), (\w -> focus w >> mouseMoveWindow w))
|
|
||||||
-- mod-button2 @@ Raise the window to the top of the stack
|
|
||||||
, ((modMask, button2), (\w -> focus w >> swapMaster))
|
|
||||||
-- mod-button3 @@ Set the window to floating mode and resize by dragging
|
|
||||||
, ((modMask, button3), (\w -> focus w >> mouseResizeWindow w)) ]
|
|
@@ -1,8 +0,0 @@
|
|||||||
module Config where
|
|
||||||
import Graphics.X11.Xlib.Types (Dimension)
|
|
||||||
import Graphics.X11.Xlib (KeyMask)
|
|
||||||
import XMonad
|
|
||||||
borderWidth :: Dimension
|
|
||||||
logHook :: X ()
|
|
||||||
numlockMask :: KeyMask
|
|
||||||
workspaces :: [WorkspaceId]
|
|
2
LICENSE
2
LICENSE
@@ -14,7 +14,7 @@ are met:
|
|||||||
may be used to endorse or promote products derived from this software
|
may be used to endorse or promote products derived from this software
|
||||||
without specific prior written permission.
|
without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND
|
||||||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||||
ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
|
ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
|
||||||
|
274
Main.hs
274
Main.hs
@@ -1,6 +1,6 @@
|
|||||||
-----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
-- |
|
-- |
|
||||||
-- Module : Main.hs
|
-- Module : Main
|
||||||
-- Copyright : (c) Spencer Janssen 2007
|
-- Copyright : (c) Spencer Janssen 2007
|
||||||
-- License : BSD3-style (see LICENSE)
|
-- License : BSD3-style (see LICENSE)
|
||||||
--
|
--
|
||||||
@@ -9,246 +9,54 @@
|
|||||||
-- Portability : not portable, uses mtl, X11, posix
|
-- Portability : not portable, uses mtl, X11, posix
|
||||||
--
|
--
|
||||||
-- xmonad, a minimalist, tiling window manager for X11
|
-- xmonad, a minimalist, tiling window manager for X11
|
||||||
--
|
--
|
||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
module Main where
|
module Main (main) where
|
||||||
|
|
||||||
import Data.Bits
|
|
||||||
import qualified Data.Map as M
|
|
||||||
import qualified Data.Set as S
|
|
||||||
import Control.Monad.Reader
|
|
||||||
import Control.Monad.State
|
|
||||||
import Data.Maybe (fromMaybe)
|
|
||||||
|
|
||||||
import System.Environment (getArgs)
|
|
||||||
|
|
||||||
import Graphics.X11.Xlib hiding (refreshKeyboardMapping)
|
|
||||||
import Graphics.X11.Xlib.Extras
|
|
||||||
import Graphics.X11.Xinerama (getScreenInfo)
|
|
||||||
|
|
||||||
import XMonad
|
import XMonad
|
||||||
import Config
|
|
||||||
import StackSet (new, floating, member, findIndex, workspace, tag, current, visible)
|
|
||||||
import qualified StackSet as W
|
|
||||||
import Operations
|
|
||||||
|
|
||||||
|
import Control.Exception (handle)
|
||||||
import System.IO
|
import System.IO
|
||||||
|
import System.Info
|
||||||
|
import System.Environment
|
||||||
|
import System.Posix.Process (executeFile)
|
||||||
|
|
||||||
-- |
|
#ifdef TESTING
|
||||||
-- The main entry point
|
import qualified Properties
|
||||||
--
|
#endif
|
||||||
|
|
||||||
|
-- | The entry point into xmonad. Attempts to compile any custom main
|
||||||
|
-- for xmonad, and if it doesn't find one, just launches the default.
|
||||||
main :: IO ()
|
main :: IO ()
|
||||||
main = do
|
main = do
|
||||||
dpy <- openDisplay ""
|
|
||||||
let dflt = defaultScreen dpy
|
|
||||||
|
|
||||||
rootw <- rootWindow dpy dflt
|
|
||||||
xinesc <- getScreenInfo dpy
|
|
||||||
nbc <- initColor dpy normalBorderColor
|
|
||||||
fbc <- initColor dpy focusedBorderColor
|
|
||||||
hSetBuffering stdout NoBuffering
|
|
||||||
args <- getArgs
|
args <- getArgs
|
||||||
|
let launch = handle (hPrint stderr) buildLaunch >> xmonad defaultConfig
|
||||||
|
case args of
|
||||||
|
[] -> launch
|
||||||
|
["--resume", _] -> launch
|
||||||
|
["--recompile"] -> recompile False >> return ()
|
||||||
|
["--recompile-force"] -> recompile True >> return ()
|
||||||
|
["--version"] -> putStrLn "xmonad 0.5"
|
||||||
|
#ifdef TESTING
|
||||||
|
("--run-tests":_) -> Properties.main
|
||||||
|
#endif
|
||||||
|
_ -> fail "unrecognized flags"
|
||||||
|
|
||||||
let winset | ("--resume" : s : _) <- args
|
-- | Build "~/.xmonad/xmonad.hs" with ghc, then execute it. If there are no
|
||||||
, [(x, "")] <- reads s = x
|
-- errors, this function does not return. An exception is raised in any of
|
||||||
| otherwise = new workspaces $ zipWith SD xinesc gaps
|
-- these cases:
|
||||||
gaps = take (length xinesc) $ defaultGaps ++ repeat (0,0,0,0)
|
-- * ghc missing
|
||||||
|
-- * ~/.xmonad/xmonad.hs missing
|
||||||
safeLayouts = case defaultLayouts of [] -> (full, []); (x:xs) -> (x,xs)
|
-- * xmonad.hs fails to compile
|
||||||
cf = XConf
|
-- ** wrong ghc in path (fails to compile)
|
||||||
{ display = dpy
|
-- ** type error, syntax error, ..
|
||||||
, theRoot = rootw
|
-- * Missing xmonad/XMonadContrib modules due to ghc upgrade
|
||||||
, normalBorder = nbc
|
|
||||||
, focusedBorder = fbc }
|
|
||||||
st = XState
|
|
||||||
{ windowset = winset
|
|
||||||
, layouts = M.fromList [(w, safeLayouts) | w <- workspaces]
|
|
||||||
, mapped = S.empty
|
|
||||||
, waitingUnmap = M.empty
|
|
||||||
, dragging = Nothing }
|
|
||||||
|
|
||||||
xSetErrorHandler -- in C, I'm too lazy to write the binding: dons
|
|
||||||
|
|
||||||
-- setup initial X environment
|
|
||||||
sync dpy False
|
|
||||||
selectInput dpy rootw $ substructureRedirectMask .|. substructureNotifyMask
|
|
||||||
.|. enterWindowMask .|. leaveWindowMask .|. structureNotifyMask
|
|
||||||
grabKeys dpy rootw
|
|
||||||
grabButtons dpy rootw
|
|
||||||
|
|
||||||
sync dpy False
|
|
||||||
|
|
||||||
ws <- scan dpy rootw -- on the resume case, will pick up new windows
|
|
||||||
allocaXEvent $ \e ->
|
|
||||||
runX cf st $ do
|
|
||||||
|
|
||||||
-- walk workspace, resetting X states/mask for windows
|
|
||||||
-- TODO, general iterators for these lists.
|
|
||||||
sequence_ [ setInitialProperties w >> reveal w
|
|
||||||
| wk <- map W.workspace (W.current winset : W.visible winset)
|
|
||||||
, w <- W.integrate' (W.stack wk) ]
|
|
||||||
|
|
||||||
sequence_ [ setInitialProperties w >> hide w
|
|
||||||
| wk <- W.hidden winset
|
|
||||||
, w <- W.integrate' (W.stack wk) ]
|
|
||||||
|
|
||||||
mapM_ manage ws -- find new windows
|
|
||||||
refresh
|
|
||||||
|
|
||||||
-- main loop, for all you HOF/recursion fans out there.
|
|
||||||
forever $ handle =<< io (nextEvent dpy e >> getEvent e)
|
|
||||||
|
|
||||||
where forever a = a >> forever a
|
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
|
||||||
-- IO stuff. Doesn't require any X state
|
|
||||||
-- Most of these things run only on startup (bar grabkeys)
|
|
||||||
|
|
||||||
-- | scan for any new windows to manage. If they're already managed,
|
|
||||||
-- this should be idempotent.
|
|
||||||
scan :: Display -> Window -> IO [Window]
|
|
||||||
scan dpy rootw = do
|
|
||||||
(_, _, ws) <- queryTree dpy rootw
|
|
||||||
filterM ok ws
|
|
||||||
-- TODO: scan for windows that are either 'IsViewable' or where WM_STATE ==
|
|
||||||
-- Iconic
|
|
||||||
where ok w = do wa <- getWindowAttributes dpy w
|
|
||||||
a <- internAtom dpy "WM_STATE" False
|
|
||||||
p <- getWindowProperty32 dpy a w
|
|
||||||
let ic = case p of
|
|
||||||
Just (3:_) -> True -- 3 for iconified
|
|
||||||
_ -> False
|
|
||||||
return $ not (wa_override_redirect wa)
|
|
||||||
&& (wa_map_state wa == waIsViewable || ic)
|
|
||||||
|
|
||||||
-- | Grab the keys back
|
|
||||||
grabKeys :: Display -> Window -> IO ()
|
|
||||||
grabKeys dpy rootw = do
|
|
||||||
ungrabKey dpy anyKey anyModifier rootw
|
|
||||||
forM_ (M.keys keys) $ \(mask,sym) -> do
|
|
||||||
kc <- keysymToKeycode dpy sym
|
|
||||||
-- "If the specified KeySym is not defined for any KeyCode,
|
|
||||||
-- XKeysymToKeycode() returns zero."
|
|
||||||
when (kc /= '\0') $ mapM_ (grab kc . (mask .|.)) extraModifiers
|
|
||||||
|
|
||||||
where grab kc m = grabKey dpy kc m rootw True grabModeAsync grabModeAsync
|
|
||||||
|
|
||||||
grabButtons :: Display -> Window -> IO ()
|
|
||||||
grabButtons dpy rootw = do
|
|
||||||
ungrabButton dpy anyButton anyModifier rootw
|
|
||||||
mapM_ (\(m,b) -> mapM_ (grab b . (m .|.)) extraModifiers) (M.keys mouseBindings)
|
|
||||||
where grab button mask = grabButton dpy button mask rootw False buttonPressMask
|
|
||||||
grabModeAsync grabModeSync none none
|
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
|
||||||
-- | Event handler. Map X events onto calls into Operations.hs, which
|
|
||||||
-- modify our internal model of the window manager state.
|
|
||||||
--
|
--
|
||||||
-- Events dwm handles that we don't:
|
buildLaunch :: IO ()
|
||||||
--
|
buildLaunch = do
|
||||||
-- [ButtonPress] = buttonpress,
|
recompile False
|
||||||
-- [Expose] = expose,
|
dir <- getXMonadDir
|
||||||
-- [PropertyNotify] = propertynotify,
|
args <- getArgs
|
||||||
--
|
executeFile (dir ++ "/xmonad-"++arch++"-"++os) False args Nothing
|
||||||
|
return ()
|
||||||
handle :: Event -> X ()
|
|
||||||
|
|
||||||
-- run window manager command
|
|
||||||
handle (KeyEvent {ev_event_type = t, ev_state = m, ev_keycode = code})
|
|
||||||
| t == keyPress = withDisplay $ \dpy -> do
|
|
||||||
s <- io $ keycodeToKeysym dpy code 0
|
|
||||||
whenJust (M.lookup (cleanMask m,s) keys) id
|
|
||||||
|
|
||||||
-- manage a new window
|
|
||||||
handle (MapRequestEvent {ev_window = w}) = withDisplay $ \dpy -> do
|
|
||||||
wa <- io $ getWindowAttributes dpy w -- ignore override windows
|
|
||||||
-- need to ignore mapping requests by managed windows not on the current workspace
|
|
||||||
managed <- isClient w
|
|
||||||
when (not (wa_override_redirect wa) && not managed) $ do manage w
|
|
||||||
|
|
||||||
-- window destroyed, unmanage it
|
|
||||||
-- window gone, unmanage it
|
|
||||||
handle (DestroyWindowEvent {ev_window = w}) = whenX (isClient w) $ unmanage w
|
|
||||||
|
|
||||||
-- We track expected unmap events in waitingUnmap. We ignore this event unless
|
|
||||||
-- it is synthetic or we are not expecting an unmap notification from a window.
|
|
||||||
handle (UnmapEvent {ev_window = w, ev_send_event = synthetic}) = whenX (isClient w) $ do
|
|
||||||
e <- gets (fromMaybe 0 . M.lookup w . waitingUnmap)
|
|
||||||
if (synthetic || e == 0)
|
|
||||||
then unmanage w
|
|
||||||
else modify (\s -> s { waitingUnmap = M.adjust pred w (waitingUnmap s) })
|
|
||||||
|
|
||||||
-- set keyboard mapping
|
|
||||||
handle e@(MappingNotifyEvent {ev_window = w}) = do
|
|
||||||
io $ refreshKeyboardMapping e
|
|
||||||
when (ev_request e == mappingKeyboard) $ withDisplay $ io . flip grabKeys w
|
|
||||||
|
|
||||||
-- handle button release, which may finish dragging.
|
|
||||||
handle e@(ButtonEvent {ev_event_type = t})
|
|
||||||
| t == buttonRelease = do
|
|
||||||
drag <- gets dragging
|
|
||||||
case drag of
|
|
||||||
-- we're done dragging and have released the mouse:
|
|
||||||
Just (_,f) -> modify (\s -> s { dragging = Nothing }) >> f
|
|
||||||
Nothing -> broadcastMessage e
|
|
||||||
|
|
||||||
-- handle motionNotify event, which may mean we are dragging.
|
|
||||||
handle e@(MotionEvent {ev_event_type = _t, ev_x = x, ev_y = y}) = do
|
|
||||||
drag <- gets dragging
|
|
||||||
case drag of
|
|
||||||
Just (d,_) -> d (fromIntegral x) (fromIntegral y) -- we're dragging
|
|
||||||
Nothing -> broadcastMessage e
|
|
||||||
|
|
||||||
-- click on an unfocused window, makes it focused on this workspace
|
|
||||||
handle e@(ButtonEvent {ev_window = w,ev_event_type = t,ev_button = b })
|
|
||||||
| t == buttonPress = do
|
|
||||||
-- If it's the root window, then it's something we
|
|
||||||
-- grabbed in grabButtons. Otherwise, it's click-to-focus.
|
|
||||||
isr <- isRoot w
|
|
||||||
if isr then whenJust (M.lookup (cleanMask (ev_state e), b) mouseBindings) ($ ev_subwindow e)
|
|
||||||
else focus w
|
|
||||||
sendMessage e -- Always send button events.
|
|
||||||
|
|
||||||
-- entered a normal window, makes this focused.
|
|
||||||
handle e@(CrossingEvent {ev_window = w, ev_event_type = t})
|
|
||||||
| t == enterNotify && ev_mode e == notifyNormal
|
|
||||||
&& ev_detail e /= notifyInferior = focus w
|
|
||||||
|
|
||||||
-- left a window, check if we need to focus root
|
|
||||||
handle e@(CrossingEvent {ev_event_type = t})
|
|
||||||
| t == leaveNotify
|
|
||||||
= do rootw <- asks theRoot
|
|
||||||
when (ev_window e == rootw && not (ev_same_screen e)) $ setFocusX rootw
|
|
||||||
|
|
||||||
-- configure a window
|
|
||||||
handle e@(ConfigureRequestEvent {ev_window = w}) = withDisplay $ \dpy -> do
|
|
||||||
ws <- gets windowset
|
|
||||||
wa <- io $ getWindowAttributes dpy w
|
|
||||||
|
|
||||||
-- TODO temporary workaround for some bugs in float. Don't call 'float' on
|
|
||||||
-- windows that aren't visible, because it changes the focused screen
|
|
||||||
let vis = any ((== findIndex w ws) . Just . tag . workspace) (current ws : visible ws)
|
|
||||||
if (M.member w (floating ws) && vis)
|
|
||||||
|| not (member w ws)
|
|
||||||
then do io $ configureWindow dpy w (ev_value_mask e) $ WindowChanges
|
|
||||||
{ wc_x = ev_x e
|
|
||||||
, wc_y = ev_y e
|
|
||||||
, wc_width = ev_width e
|
|
||||||
, wc_height = ev_height e
|
|
||||||
, wc_border_width = fromIntegral borderWidth
|
|
||||||
, wc_sibling = ev_above e
|
|
||||||
, wc_stack_mode = ev_detail e }
|
|
||||||
when (member w ws) (float w)
|
|
||||||
else io $ allocaXEvent $ \ev -> do
|
|
||||||
setEventType ev configureNotify
|
|
||||||
setConfigureEvent ev w w
|
|
||||||
(wa_x wa) (wa_y wa) (wa_width wa)
|
|
||||||
(wa_height wa) (ev_border_width e) none (wa_override_redirect wa)
|
|
||||||
sendEvent dpy w False 0 ev
|
|
||||||
io $ sync dpy False
|
|
||||||
|
|
||||||
-- the root may have configured
|
|
||||||
handle (ConfigureEvent {ev_window = w}) = whenX (isRoot w) rescreen
|
|
||||||
|
|
||||||
handle e = broadcastMessage e -- trace (eventName e) -- ignoring
|
|
||||||
|
160
README
160
README
@@ -1,70 +1,101 @@
|
|||||||
xmonad : a lightweight X11 window manager.
|
xmonad : a tiling window manager
|
||||||
|
|
||||||
http://xmonad.org
|
http://xmonad.org
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
xmonad is a tiling window manager for X. Windows are arranged
|
||||||
|
automatically to tile the screen without gaps or overlap, maximising
|
||||||
About:
|
screen use. Window manager features are accessible from the
|
||||||
|
keyboard: a mouse is optional. xmonad is written, configured and
|
||||||
Xmonad is a tiling window manager for X. Windows are managed using
|
extensible in Haskell. Custom layout algorithms, key bindings and
|
||||||
automatic tiling algorithms, which can be dynamically configured.
|
other extensions may be written by the user in config files. Layouts
|
||||||
Windows are arranged so as to tile the screen without gaps, maximising
|
are applied dynamically, and different layouts may be used on each
|
||||||
screen use. All features of the window manager are accessible
|
workspace. Xinerama is fully supported, allowing windows to be tiled
|
||||||
from the keyboard: a mouse is strictly optional. Xmonad is written
|
on several physical screens.
|
||||||
and extensible in Haskell, and custom layout algorithms may be
|
|
||||||
implemented by the user in config files. A guiding principle of the
|
|
||||||
user interface is <i>predictability</i>: users should know in
|
|
||||||
advance precisely the window arrangement that will result from any
|
|
||||||
action, leading to an intuitive user interface.
|
|
||||||
|
|
||||||
Xmonad provides three tiling algorithms by default: tall, wide and
|
|
||||||
fullscreen. In tall or wide mode, all windows are visible and tiled
|
|
||||||
to fill the plane without gaps. In fullscreen mode only the focused
|
|
||||||
window is visible, filling the screen. Alternative tiling
|
|
||||||
algorithms are provided as extensions. Sets of windows are grouped
|
|
||||||
together on virtual workspaces and each workspace retains its own
|
|
||||||
layout. Multiple physical monitors are supported via Xinerama,
|
|
||||||
allowing simultaneous display of several workspaces.
|
|
||||||
|
|
||||||
Adhering to a minimalist philosophy of doing one job, and doing it
|
|
||||||
well, the entire code base remains tiny, and is written to be simple
|
|
||||||
to understand and modify. By using Haskell as a configuration
|
|
||||||
language arbitrarily complex extensions may be implemented by the
|
|
||||||
user using a powerful `scripting' language, without needing to
|
|
||||||
modify the window manager directly. For example, users may write
|
|
||||||
their own tiling algorithms.
|
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
|
||||||
|
|
||||||
Building:
|
Building:
|
||||||
|
|
||||||
Get the dependencies
|
Building is quite straightforward, and requries a basic Haskell toolchain.
|
||||||
|
On many systems xmonad is available as a binary package in your
|
||||||
|
package system (e.g. on debian or gentoo). If at all possible, use this
|
||||||
|
in preference to a source build, as the dependency resolution will be
|
||||||
|
simpler.
|
||||||
|
|
||||||
It is likely that you already have some of these dependencies. To check
|
We'll now walk through the complete list of toolchain dependencies.
|
||||||
whether you've got a package run 'ghc-pkg list some_package_name'
|
|
||||||
|
|
||||||
mtl http://hackage.haskell.org/cgi-bin/hackage-scripts/package/mtl-1.0
|
* GHC: the Glasgow Haskell Compiler
|
||||||
unix http://hackage.haskell.org/cgi-bin/hackage-scripts/package/unix-2.0
|
|
||||||
X11 http://hackage.haskell.org/cgi-bin/hackage-scripts/package/X11-1.2.2
|
You first need a Haskell compiler. Your distribution's package
|
||||||
X11-extras: http://hackage.haskell.org/cgi-bin/hackage-scripts/package/X11-extras-0.3
|
system will have binaries of GHC (the Glasgow Haskell Compiler), the
|
||||||
|
compiler we use, so install that first. If your operating system's
|
||||||
|
package system doesn't provide a binary version of GHC, you can find
|
||||||
|
them here:
|
||||||
|
|
||||||
And then build with Cabal:
|
http://haskell.org/ghc
|
||||||
|
|
||||||
runhaskell Setup.lhs configure --prefix=$HOME
|
For example, in Debian you would install GHC with:
|
||||||
runhaskell Setup.lhs build
|
|
||||||
runhaskell Setup.lhs install --user
|
apt-get install ghc6
|
||||||
|
|
||||||
|
It shouldn't be necessary to compile GHC from source -- every common
|
||||||
|
system has a pre-build binary version.
|
||||||
|
|
||||||
|
* X11 libraries:
|
||||||
|
|
||||||
|
Since you're building an X application, you'll need the C X11
|
||||||
|
library headers. On many platforms, these come pre-installed. For
|
||||||
|
others, such as Debian, you can get them from your package manager:
|
||||||
|
|
||||||
|
apt-get install libx11-dev
|
||||||
|
|
||||||
|
Typically you need: libXinerama libXext libX11
|
||||||
|
|
||||||
|
* Cabal
|
||||||
|
|
||||||
|
xmonad requires a recent version of Cabal, >= 1.2.0. If you're using
|
||||||
|
GHC 6.8, then it comes bundled with the right version. If you're
|
||||||
|
using GHC 6.6.x, you'll need to build and install Cabal from hackage
|
||||||
|
first:
|
||||||
|
|
||||||
|
http://hackage.haskell.org/cgi-bin/hackage-scripts/package/Cabal
|
||||||
|
|
||||||
|
You can check which version you have with the command:
|
||||||
|
|
||||||
|
$ ghc-pkg list Cabal
|
||||||
|
Cabal-1.2.2.0
|
||||||
|
|
||||||
|
* Haskell libraries: mtl, unix, X11
|
||||||
|
|
||||||
|
Finally, you need the Haskell libraries xmonad depends on. Since
|
||||||
|
you've a working GHC installation now, most of these will be
|
||||||
|
provided. To check whether you've got a package run 'ghc-pkg list
|
||||||
|
some_package_name'. You will need the following packages:
|
||||||
|
|
||||||
|
mtl http://hackage.haskell.org/cgi-bin/hackage-scripts/package/mtl-1.0
|
||||||
|
unix http://hackage.haskell.org/cgi-bin/hackage-scripts/package/unix-2.0
|
||||||
|
X11 http://hackage.haskell.org/cgi-bin/hackage-scripts/package/X11-1.4.1
|
||||||
|
|
||||||
|
* Build xmonad:
|
||||||
|
|
||||||
|
Once you've got all the dependencies in place (which should be
|
||||||
|
straightforward), build xmonad:
|
||||||
|
|
||||||
|
runhaskell Setup.lhs configure --user --prefix=$HOME
|
||||||
|
runhaskell Setup.lhs build
|
||||||
|
runhaskell Setup.lhs install --user
|
||||||
|
|
||||||
|
And you're done!
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
Notes for using the darcs version
|
Notes for using the darcs version
|
||||||
|
|
||||||
If you're building the darcs version of xmonad, be sure to also
|
If you're building the darcs version of xmonad, be sure to also
|
||||||
use the darcs version of X11-extras, which is developed concurrently
|
use the darcs version of the X11 library, which is developed
|
||||||
with xmonad.
|
concurrently with xmonad.
|
||||||
|
|
||||||
darcs get http://darcs.haskell.org/~sjanssen/X11-extras
|
darcs get http://darcs.haskell.org/X11
|
||||||
|
|
||||||
Not using X11-extras from darcs, is the most common reason for the
|
Not using X11 from darcs is the most common reason for the
|
||||||
darcs version of xmonad to fail to build.
|
darcs version of xmonad to fail to build.
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
@@ -79,33 +110,40 @@ Running xmonad:
|
|||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
Configuring:
|
||||||
|
|
||||||
|
See the CONFIG document
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
XMonadContrib
|
XMonadContrib
|
||||||
|
|
||||||
There are various contributed modules that can be used with xmonad.
|
There are many extensions to xmonad available in the XMonadContrib
|
||||||
Examples include an ion3-like tabbed layout, a prompt/program launcher,
|
(xmc) library. Examples include an ion3-like tabbed layout, a
|
||||||
and various other useful modules. XMonadContrib is available at:
|
prompt/program launcher, and various other useful modules.
|
||||||
|
XMonadContrib is available at:
|
||||||
|
|
||||||
0.3 release: http://www.xmonad.org/XMonadContrib-0.3.tar.gz
|
0.5 release: http://hackage.haskell.org/cgi-bin/hackage-scripts/package/xmonad-contrib-0.5
|
||||||
|
|
||||||
darcs version: darcs get http://darcs.haskell.org/~sjanssen/XMonadContrib
|
darcs version: darcs get http://code.haskell.org/XMonadContrib
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
Other useful programs:
|
Other useful programs:
|
||||||
|
|
||||||
For a program dispatch menu:
|
A nicer xterm replacment, that supports resizing better:
|
||||||
|
|
||||||
dmenu http://www.suckless.org/download/
|
urxvt http://software.schmorp.de/pkg/rxvt-unicode.html
|
||||||
or
|
|
||||||
gmrun (in your package system)
|
|
||||||
|
|
||||||
For custom status bars:
|
For custom status bars:
|
||||||
|
|
||||||
dzen http://gotmor.googlepages.com/dzen
|
dzen http://gotmor.googlepages.com/dzen
|
||||||
|
xmobar http://hackage.haskell.org/cgi-bin/hackage-scripts/package/xmobar
|
||||||
|
|
||||||
A nicer xterm replacment, that supports resizing better:
|
For a program dispatch menu:
|
||||||
|
|
||||||
urxvt http://software.schmorp.de/pkg/rxvt-unicode.html
|
dmenu http://www.suckless.org/download/
|
||||||
|
gmrun (in your package system)
|
||||||
|
|
||||||
Authors:
|
Authors:
|
||||||
|
|
||||||
|
21
STYLE
Normal file
21
STYLE
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
|
||||||
|
== Coding guidelines for contributing to
|
||||||
|
== xmonad and the xmonad contributed extensions
|
||||||
|
|
||||||
|
* Comment every top level function (particularly exported funtions), and
|
||||||
|
provide a type signature; use Haddock syntax in the comments.
|
||||||
|
|
||||||
|
* Follow the coding style of the other modules.
|
||||||
|
|
||||||
|
* Code should be compilable with -Wall -Werror. There should be no warnings.
|
||||||
|
|
||||||
|
* Partial functions should be avoided: the window manager should not
|
||||||
|
crash, so do not call `error` or `undefined`
|
||||||
|
|
||||||
|
* Tabs are illegal. Use 4 spaces for indenting.
|
||||||
|
|
||||||
|
* Any pure function added to the core should have QuickCheck properties
|
||||||
|
precisely defining its behaviour.
|
||||||
|
|
||||||
|
* New modules should identify the author, and be submitted under
|
||||||
|
the same license as xmonad (BSD3 license or freer).
|
29
TODO
29
TODO
@@ -1,17 +1,20 @@
|
|||||||
0.3 release:
|
- Write down invariants for the window life cycle, especially:
|
||||||
* stable contrib repo tarball
|
- When are borders set? Prove that the current handling is sufficient.
|
||||||
* haddocks for core and contribs on xmonad.org
|
|
||||||
* tag xmonad
|
|
||||||
* tag X11-extras
|
|
||||||
* tag X11
|
|
||||||
* more QC tests
|
|
||||||
|
|
||||||
|
- current floating layer handling is unoptimal. FocusUp should raise,
|
||||||
|
for example
|
||||||
|
|
||||||
- possibles:
|
- Issues still with stacking order.
|
||||||
- use more constrained type in StackSet to avoid pattern match warnings
|
|
||||||
- audit for events handled in dwm.
|
|
||||||
|
|
||||||
- related:
|
= Release management =
|
||||||
- xcb bindings
|
|
||||||
- randr
|
|
||||||
|
|
||||||
|
* configuration documentation
|
||||||
|
|
||||||
|
* generate haddocks for core and XMC, upload to xmonad.org
|
||||||
|
* generate manpage, generate html manpage
|
||||||
|
* double check README build instructions
|
||||||
|
* test core with 6.6 and 6.8
|
||||||
|
* bump xmonad.cabal version and X11 version
|
||||||
|
* upload X11 and xmonad to hackage
|
||||||
|
* check examples/text in user-facing Config.hs
|
||||||
|
* check tour.html and intro.html are up to date, and mention all core bindings
|
||||||
|
239
XMonad.hs
239
XMonad.hs
@@ -1,212 +1,47 @@
|
|||||||
{-# OPTIONS -fglasgow-exts #-}
|
--------------------------------------------------------------------
|
||||||
-----------------------------------------------------------------------------
|
|
||||||
-- |
|
-- |
|
||||||
-- Module : XMonad.hs
|
-- Module : XMonad
|
||||||
-- Copyright : (c) Spencer Janssen 2007
|
-- Copyright : (c) Don Stewart
|
||||||
-- License : BSD3-style (see LICENSE)
|
-- License : BSD3
|
||||||
--
|
--
|
||||||
-- Maintainer : sjanssen@cse.unl.edu
|
-- Maintainer: Don Stewart <dons@galois.com>
|
||||||
-- Stability : unstable
|
-- Stability : provisional
|
||||||
-- Portability : not portable, uses cunning newtype deriving
|
-- Portability:
|
||||||
--
|
--
|
||||||
-- The X monad, a state monad transformer over IO, for the window
|
--------------------------------------------------------------------
|
||||||
-- manager state, and support routines.
|
|
||||||
--
|
--
|
||||||
-----------------------------------------------------------------------------
|
-- Useful exports for configuration files.
|
||||||
|
|
||||||
module XMonad (
|
module XMonad (
|
||||||
X, WindowSet, WorkspaceId(..), ScreenId(..), ScreenDetail(..), XState(..), XConf(..), Layout(..),
|
|
||||||
Typeable, Message, SomeMessage(..), fromMessage, runLayout,
|
|
||||||
runX, catchX, io, catchIO, withDisplay, withWindowSet, isRoot, spawn, restart, trace, whenJust, whenX,
|
|
||||||
atom_WM_STATE, atom_WM_PROTOCOLS, atom_WM_DELETE_WINDOW
|
|
||||||
) where
|
|
||||||
|
|
||||||
import StackSet
|
module XMonad.Main,
|
||||||
|
module XMonad.Core,
|
||||||
|
module XMonad.Config,
|
||||||
|
module XMonad.Layout,
|
||||||
|
module XMonad.ManageHook,
|
||||||
|
module XMonad.Operations,
|
||||||
|
module Graphics.X11,
|
||||||
|
module Graphics.X11.Xlib.Extras,
|
||||||
|
(.|.),
|
||||||
|
MonadState(..), gets, modify,
|
||||||
|
MonadReader(..), asks,
|
||||||
|
MonadIO(..)
|
||||||
|
|
||||||
|
) where
|
||||||
|
|
||||||
|
-- core modules
|
||||||
|
import XMonad.Main
|
||||||
|
import XMonad.Core
|
||||||
|
import XMonad.Config
|
||||||
|
import XMonad.Layout
|
||||||
|
import XMonad.ManageHook
|
||||||
|
import XMonad.Operations
|
||||||
|
-- import XMonad.StackSet -- conflicts with 'workspaces' defined in XMonad.hs
|
||||||
|
|
||||||
|
-- modules needed to get basic configuration working
|
||||||
|
import Data.Bits
|
||||||
|
import Graphics.X11 hiding (refreshKeyboardMapping)
|
||||||
|
import Graphics.X11.Xlib.Extras
|
||||||
|
|
||||||
import Control.Monad.State
|
import Control.Monad.State
|
||||||
import Control.Monad.Reader
|
import Control.Monad.Reader
|
||||||
import System.IO
|
|
||||||
import System.Posix.Process (executeFile, forkProcess, getProcessStatus, createSession)
|
|
||||||
import System.Exit
|
|
||||||
import System.Environment
|
|
||||||
import Graphics.X11.Xlib
|
|
||||||
-- for Read instance
|
|
||||||
import Graphics.X11.Xlib.Extras ()
|
|
||||||
import Data.Typeable
|
|
||||||
|
|
||||||
import qualified Data.Map as M
|
|
||||||
import qualified Data.Set as S
|
|
||||||
|
|
||||||
-- | XState, the window manager state.
|
|
||||||
-- Just the display, width, height and a window list
|
|
||||||
data XState = XState
|
|
||||||
{ windowset :: !WindowSet -- ^ workspace list
|
|
||||||
, mapped :: !(S.Set Window) -- ^ the Set of mapped windows
|
|
||||||
, waitingUnmap :: !(M.Map Window Int) -- ^ the number of expected UnmapEvents
|
|
||||||
, layouts :: !(M.Map WorkspaceId (Layout Window, [Layout Window]))
|
|
||||||
-- ^ mapping of workspaces to descriptions of their layouts
|
|
||||||
, dragging :: !(Maybe (Position -> Position -> X (), X ())) }
|
|
||||||
data XConf = XConf
|
|
||||||
{ display :: Display -- ^ the X11 display
|
|
||||||
, theRoot :: !Window -- ^ the root window
|
|
||||||
, normalBorder :: !Pixel -- ^ border color of unfocused windows
|
|
||||||
, focusedBorder :: !Pixel } -- ^ border color of the focused window
|
|
||||||
|
|
||||||
type WindowSet = StackSet WorkspaceId Window ScreenId ScreenDetail
|
|
||||||
|
|
||||||
-- | Virtual workspace indicies
|
|
||||||
newtype WorkspaceId = W Int deriving (Eq,Ord,Show,Read,Enum,Num,Integral,Real)
|
|
||||||
|
|
||||||
-- | Physical screen indicies
|
|
||||||
newtype ScreenId = S Int deriving (Eq,Ord,Show,Read,Enum,Num,Integral,Real)
|
|
||||||
|
|
||||||
data ScreenDetail = SD { screenRect :: !Rectangle
|
|
||||||
, statusGap :: !(Int,Int,Int,Int) -- ^ width of status bar on the screen
|
|
||||||
} deriving (Eq,Show, Read)
|
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
|
||||||
|
|
||||||
-- | 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 ()
|
|
||||||
|
|
||||||
-- | Run in the X monad, and in case of exception, and catch it and log it
|
|
||||||
-- to stderr, and run the error case.
|
|
||||||
catchX :: X a -> X a -> X a
|
|
||||||
catchX (X job) (X errcase) = do
|
|
||||||
st <- get
|
|
||||||
c <- ask
|
|
||||||
(a,s') <- io ((runStateT (runReaderT job c) st) `catch`
|
|
||||||
\e -> (do hPutStrLn stderr (show e); runStateT (runReaderT errcase c) st))
|
|
||||||
put s'
|
|
||||||
return a
|
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
|
||||||
-- Convenient wrappers to state
|
|
||||||
|
|
||||||
-- | Run a monad action with the current display settings
|
|
||||||
withDisplay :: (Display -> X a) -> X a
|
|
||||||
withDisplay f = asks display >>= f
|
|
||||||
|
|
||||||
-- | Run a monadic action with the current stack set
|
|
||||||
withWindowSet :: (WindowSet -> X a) -> X a
|
|
||||||
withWindowSet f = gets windowset >>= f
|
|
||||||
|
|
||||||
-- | True if the given window is the root window
|
|
||||||
isRoot :: Window -> X Bool
|
|
||||||
isRoot w = liftM (w==) (asks theRoot)
|
|
||||||
|
|
||||||
-- | Wrapper for the common case of atom internment
|
|
||||||
getAtom :: String -> X Atom
|
|
||||||
getAtom str = withDisplay $ \dpy -> io $ internAtom dpy str False
|
|
||||||
|
|
||||||
-- | Common non-predefined atoms
|
|
||||||
atom_WM_PROTOCOLS, atom_WM_DELETE_WINDOW, atom_WM_STATE :: X Atom
|
|
||||||
atom_WM_PROTOCOLS = getAtom "WM_PROTOCOLS"
|
|
||||||
atom_WM_DELETE_WINDOW = getAtom "WM_DELETE_WINDOW"
|
|
||||||
atom_WM_STATE = getAtom "WM_STATE"
|
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
|
||||||
-- | Layout handling
|
|
||||||
|
|
||||||
-- The different layout modes
|
|
||||||
-- 'doLayout': given a Rectangle and a Stack, layout the stack elements
|
|
||||||
-- inside the given Rectangle. If an element is not given a Rectangle
|
|
||||||
-- by 'doLayout', then it is not shown on screen. Windows are restacked
|
|
||||||
-- according to the order they are returned by 'doLayout'.
|
|
||||||
--
|
|
||||||
-- 'modifyLayout' performs message handling for that layout. If
|
|
||||||
-- 'modifyLayout' returns Nothing, then the layout did not respond to
|
|
||||||
-- that message and the screen is not refreshed. Otherwise, 'modifyLayout'
|
|
||||||
-- returns an updated 'Layout' and the screen is refreshed.
|
|
||||||
--
|
|
||||||
data Layout a = Layout { doLayout :: Rectangle -> Stack a -> X ([(a, Rectangle)], Maybe (Layout a))
|
|
||||||
, modifyLayout :: SomeMessage -> X (Maybe (Layout a)) }
|
|
||||||
|
|
||||||
runLayout :: Layout a -> Rectangle -> StackOrNot a -> X ([(a, Rectangle)], Maybe (Layout a))
|
|
||||||
runLayout l r = maybe (return ([], Nothing)) (doLayout l r)
|
|
||||||
|
|
||||||
-- | 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
|
|
||||||
|
|
||||||
-- | Lift an IO action into the X monad. If the action results in an IO
|
|
||||||
-- exception, log the exception to stderr and continue normal execution.
|
|
||||||
catchIO :: IO () -> X ()
|
|
||||||
catchIO f = liftIO (f `catch` \e -> do hPutStrLn stderr (show e); hFlush stderr)
|
|
||||||
|
|
||||||
-- | 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
|
|
||||||
getProcessStatus True False pid
|
|
||||||
return ()
|
|
||||||
|
|
||||||
-- | Restart xmonad via exec().
|
|
||||||
--
|
|
||||||
-- If the first parameter is 'Just name', restart will attempt to execute the
|
|
||||||
-- program corresponding to 'name'. Otherwise, xmonad will attempt to execute
|
|
||||||
-- the name of the current program.
|
|
||||||
--
|
|
||||||
-- When the second parameter is 'True', xmonad will attempt to resume with the
|
|
||||||
-- current window state.
|
|
||||||
restart :: Maybe String -> Bool -> X ()
|
|
||||||
restart mprog resume = do
|
|
||||||
prog <- maybe (io getProgName) return mprog
|
|
||||||
args <- if resume then gets (("--resume":) . return . show . windowset) else return []
|
|
||||||
catchIO (executeFile prog 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
|
|
||||||
|
261
XMonad/Config.hs
Normal file
261
XMonad/Config.hs
Normal file
@@ -0,0 +1,261 @@
|
|||||||
|
{-# OPTIONS -fno-warn-missing-signatures #-}
|
||||||
|
-----------------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : XMonad.Config
|
||||||
|
-- Copyright : (c) Spencer Janssen 2007
|
||||||
|
-- License : BSD3-style (see LICENSE)
|
||||||
|
--
|
||||||
|
-- Maintainer : dons@galois.com
|
||||||
|
-- Stability : stable
|
||||||
|
-- Portability : portable
|
||||||
|
--
|
||||||
|
-- This module specifies the default configuration values for xmonad.
|
||||||
|
--
|
||||||
|
-- DO NOT MODIFY THIS FILE! It won't work. You may configure xmonad
|
||||||
|
-- by providing your own @~\/.xmonad\/xmonad.hs@ that overrides
|
||||||
|
-- specific fields in 'defaultConfig'. For a starting point, you can
|
||||||
|
-- copy the @xmonad.hs@ found in the @man@ directory, or look at
|
||||||
|
-- examples on the xmonad wiki.
|
||||||
|
--
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
module XMonad.Config (defaultConfig) where
|
||||||
|
|
||||||
|
--
|
||||||
|
-- Useful imports
|
||||||
|
--
|
||||||
|
import XMonad.Core as XMonad hiding
|
||||||
|
(workspaces,manageHook,numlockMask,keys,logHook,borderWidth,mouseBindings
|
||||||
|
,defaultGaps,layoutHook,modMask,terminal,normalBorderColor,focusedBorderColor
|
||||||
|
,focusFollowsMouse)
|
||||||
|
import qualified XMonad.Core as XMonad
|
||||||
|
(workspaces,manageHook,numlockMask,keys,logHook,borderWidth,mouseBindings
|
||||||
|
,defaultGaps,layoutHook,modMask,terminal,normalBorderColor,focusedBorderColor
|
||||||
|
,focusFollowsMouse)
|
||||||
|
|
||||||
|
import XMonad.Layout
|
||||||
|
import XMonad.Operations
|
||||||
|
import XMonad.ManageHook
|
||||||
|
import qualified XMonad.StackSet as W
|
||||||
|
import Data.Bits ((.|.))
|
||||||
|
import qualified Data.Map as M
|
||||||
|
import System.Exit
|
||||||
|
import Graphics.X11.Xlib
|
||||||
|
|
||||||
|
-- | The default number of workspaces (virtual screens) and their names.
|
||||||
|
-- By default we use numeric strings, but any string may be used as a
|
||||||
|
-- workspace name. The number of workspaces is determined by the length
|
||||||
|
-- of this list.
|
||||||
|
--
|
||||||
|
-- A tagging example:
|
||||||
|
--
|
||||||
|
-- > workspaces = ["web", "irc", "code" ] ++ map show [4..9]
|
||||||
|
--
|
||||||
|
workspaces :: [WorkspaceId]
|
||||||
|
workspaces = map show [1 .. 9 :: Int]
|
||||||
|
|
||||||
|
-- | modMask lets you specify which modkey you want to use. The default
|
||||||
|
-- is mod1Mask ("left alt"). You may also consider using mod3Mask
|
||||||
|
-- ("right alt"), which does not conflict with emacs keybindings. The
|
||||||
|
-- "windows key" is usually mod4Mask.
|
||||||
|
--
|
||||||
|
defaultModMask :: KeyMask
|
||||||
|
defaultModMask = mod1Mask
|
||||||
|
|
||||||
|
-- | The mask for the numlock key. Numlock status is "masked" from the
|
||||||
|
-- current modifier status, so the keybindings will work with numlock on or
|
||||||
|
-- off. You may need to change this on some systems.
|
||||||
|
--
|
||||||
|
-- You can find the numlock modifier by running "xmodmap" and looking for a
|
||||||
|
-- modifier with Num_Lock bound to it:
|
||||||
|
--
|
||||||
|
-- > $ xmodmap | grep Num
|
||||||
|
-- > mod2 Num_Lock (0x4d)
|
||||||
|
--
|
||||||
|
-- Set numlockMask = 0 if you don't have a numlock key, or want to treat
|
||||||
|
-- numlock status separately.
|
||||||
|
--
|
||||||
|
numlockMask :: KeyMask
|
||||||
|
numlockMask = mod2Mask
|
||||||
|
|
||||||
|
-- | Width of the window border in pixels.
|
||||||
|
--
|
||||||
|
borderWidth :: Dimension
|
||||||
|
borderWidth = 1
|
||||||
|
|
||||||
|
-- | Border colors for unfocused and focused windows, respectively.
|
||||||
|
--
|
||||||
|
normalBorderColor, focusedBorderColor :: String
|
||||||
|
normalBorderColor = "#dddddd"
|
||||||
|
focusedBorderColor = "#ff0000"
|
||||||
|
|
||||||
|
-- | Default offset of drawable screen boundaries from each physical
|
||||||
|
-- screen. Anything non-zero here will leave a gap of that many pixels
|
||||||
|
-- on the given edge, on the that screen. A useful gap at top of screen
|
||||||
|
-- for a menu bar (e.g. 15)
|
||||||
|
--
|
||||||
|
-- An example, to set a top gap on monitor 1, and a gap on the bottom of
|
||||||
|
-- monitor 2, you'd use a list of geometries like so:
|
||||||
|
--
|
||||||
|
-- > defaultGaps = [(18,0,0,0),(0,18,0,0)] -- 2 gaps on 2 monitors
|
||||||
|
--
|
||||||
|
-- Fields are: top, bottom, left, right.
|
||||||
|
--
|
||||||
|
defaultGaps :: [(Int,Int,Int,Int)]
|
||||||
|
defaultGaps = [(0,0,0,0)] -- 15 for default dzen font
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Window rules
|
||||||
|
|
||||||
|
-- | Execute arbitrary actions and WindowSet manipulations when managing
|
||||||
|
-- a new window. You can use this to, for example, always float a
|
||||||
|
-- particular program, or have a client always appear on a particular
|
||||||
|
-- workspace.
|
||||||
|
--
|
||||||
|
-- To find the property name associated with a program, use
|
||||||
|
-- xprop | grep WM_CLASS
|
||||||
|
-- and click on the client you're interested in.
|
||||||
|
--
|
||||||
|
manageHook :: ManageHook
|
||||||
|
manageHook = composeAll
|
||||||
|
[ className =? "MPlayer" --> doFloat
|
||||||
|
, className =? "Gimp" --> doFloat ]
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Logging
|
||||||
|
|
||||||
|
-- | Perform an arbitrary action on each internal state change or X event.
|
||||||
|
-- Examples include:
|
||||||
|
-- * do nothing
|
||||||
|
-- * log the state to stdout
|
||||||
|
--
|
||||||
|
-- See the 'DynamicLog' extension for examples.
|
||||||
|
--
|
||||||
|
logHook :: X ()
|
||||||
|
logHook = return ()
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Extensible layouts
|
||||||
|
--
|
||||||
|
-- You can specify and transform your layouts by modifying these values.
|
||||||
|
-- If you change layout bindings be sure to use 'mod-shift-space' after
|
||||||
|
-- restarting (with 'mod-q') to reset your layout state to the new
|
||||||
|
-- defaults, as xmonad preserves your old layout settings by default.
|
||||||
|
--
|
||||||
|
|
||||||
|
-- | The available layouts. Note that each layout is separated by |||, which
|
||||||
|
-- denotes layout choice.
|
||||||
|
layout = tiled ||| Mirror tiled ||| Full
|
||||||
|
where
|
||||||
|
-- default tiling algorithm partitions the screen into two panes
|
||||||
|
tiled = Tall nmaster delta ratio
|
||||||
|
|
||||||
|
-- The default number of windows in the master pane
|
||||||
|
nmaster = 1
|
||||||
|
|
||||||
|
-- Default proportion of screen occupied by master pane
|
||||||
|
ratio = 1/2
|
||||||
|
|
||||||
|
-- Percent of screen to increment by when resizing panes
|
||||||
|
delta = 3/100
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Key bindings:
|
||||||
|
|
||||||
|
-- | The preferred terminal program, which is used in a binding below and by
|
||||||
|
-- certain contrib modules.
|
||||||
|
terminal :: String
|
||||||
|
terminal = "xterm"
|
||||||
|
|
||||||
|
-- | Whether focus follows the mouse pointer.
|
||||||
|
focusFollowsMouse :: Bool
|
||||||
|
focusFollowsMouse = True
|
||||||
|
|
||||||
|
-- | The xmonad key bindings. Add, modify or remove key bindings here.
|
||||||
|
--
|
||||||
|
-- (The comment formatting character is used when generating the manpage)
|
||||||
|
--
|
||||||
|
keys :: XConfig Layout -> M.Map (KeyMask, KeySym) (X ())
|
||||||
|
keys conf@(XConfig {XMonad.modMask = modMask}) = M.fromList $
|
||||||
|
-- launching and killing programs
|
||||||
|
[ ((modMask .|. shiftMask, xK_Return), spawn $ XMonad.terminal conf) -- %! Launch terminal
|
||||||
|
, ((modMask, xK_p ), spawn "exe=`dmenu_path | dmenu` && eval \"exec $exe\"") -- %! Launch dmenu
|
||||||
|
, ((modMask .|. shiftMask, xK_p ), spawn "gmrun") -- %! Launch gmrun
|
||||||
|
, ((modMask .|. shiftMask, xK_c ), kill) -- %! Close the focused window
|
||||||
|
|
||||||
|
, ((modMask, xK_space ), sendMessage NextLayout) -- %! Rotate through the available layout algorithms
|
||||||
|
, ((modMask .|. shiftMask, xK_space ), setLayout $ XMonad.layoutHook conf) -- %! Reset the layouts on the current workspace to default
|
||||||
|
|
||||||
|
, ((modMask, xK_n ), refresh) -- %! Resize viewed windows to the correct size
|
||||||
|
|
||||||
|
-- move focus up or down the window stack
|
||||||
|
, ((modMask, xK_Tab ), windows W.focusDown) -- %! Move focus to the next window
|
||||||
|
, ((modMask, xK_j ), windows W.focusDown) -- %! Move focus to the next window
|
||||||
|
, ((modMask, xK_k ), windows W.focusUp ) -- %! Move focus to the previous window
|
||||||
|
, ((modMask, xK_m ), windows W.focusMaster ) -- %! Move focus to the master window
|
||||||
|
|
||||||
|
-- modifying the window order
|
||||||
|
, ((modMask, xK_Return), windows W.swapMaster) -- %! Swap the focused window and the master window
|
||||||
|
, ((modMask .|. shiftMask, xK_j ), windows W.swapDown ) -- %! Swap the focused window with the next window
|
||||||
|
, ((modMask .|. shiftMask, xK_k ), windows W.swapUp ) -- %! Swap the focused window with the previous window
|
||||||
|
|
||||||
|
-- resizing the master/slave ratio
|
||||||
|
, ((modMask, xK_h ), sendMessage Shrink) -- %! Shrink the master area
|
||||||
|
, ((modMask, xK_l ), sendMessage Expand) -- %! Expand the master area
|
||||||
|
|
||||||
|
-- floating layer support
|
||||||
|
, ((modMask, xK_t ), withFocused $ windows . W.sink) -- %! Push window back into tiling
|
||||||
|
|
||||||
|
-- increase or decrease number of windows in the master area
|
||||||
|
, ((modMask , xK_comma ), sendMessage (IncMasterN 1)) -- %! Increment the number of windows in the master area
|
||||||
|
, ((modMask , xK_period), sendMessage (IncMasterN (-1))) -- %! Deincrement the number of windows in the master area
|
||||||
|
|
||||||
|
-- toggle the status bar gap
|
||||||
|
, ((modMask , xK_b ), modifyGap (\i n -> let x = (XMonad.defaultGaps conf ++ repeat (0,0,0,0)) !! i in if n == x then (0,0,0,0) else x)) -- %! Toggle the status bar gap
|
||||||
|
|
||||||
|
-- quit, or restart
|
||||||
|
, ((modMask .|. shiftMask, xK_q ), io (exitWith ExitSuccess)) -- %! Quit xmonad
|
||||||
|
, ((modMask , xK_q ), restart "xmonad" True) -- %! Restart xmonad
|
||||||
|
]
|
||||||
|
++
|
||||||
|
-- mod-[1..9] %! Switch to workspace N
|
||||||
|
-- mod-shift-[1..9] %! Move client to workspace N
|
||||||
|
[((m .|. modMask, k), windows $ f i)
|
||||||
|
| (i, k) <- zip (XMonad.workspaces conf) [xK_1 .. xK_9]
|
||||||
|
, (f, m) <- [(W.greedyView, 0), (W.shift, shiftMask)]]
|
||||||
|
++
|
||||||
|
-- mod-{w,e,r} %! Switch to physical/Xinerama screens 1, 2, or 3
|
||||||
|
-- mod-shift-{w,e,r} %! Move client to screen 1, 2, or 3
|
||||||
|
[((m .|. modMask, key), screenWorkspace sc >>= flip whenJust (windows . f))
|
||||||
|
| (key, sc) <- zip [xK_w, xK_e, xK_r] [0..]
|
||||||
|
, (f, m) <- [(W.view, 0), (W.shift, shiftMask)]]
|
||||||
|
|
||||||
|
-- | Mouse bindings: default actions bound to mouse events
|
||||||
|
--
|
||||||
|
mouseBindings :: XConfig Layout -> M.Map (KeyMask, Button) (Window -> X ())
|
||||||
|
mouseBindings (XConfig {XMonad.modMask = modMask}) = M.fromList $
|
||||||
|
-- mod-button1 %! Set the window to floating mode and move by dragging
|
||||||
|
[ ((modMask, button1), (\w -> focus w >> mouseMoveWindow w))
|
||||||
|
-- mod-button2 %! Raise the window to the top of the stack
|
||||||
|
, ((modMask, button2), (\w -> focus w >> windows W.swapMaster))
|
||||||
|
-- mod-button3 %! Set the window to floating mode and resize by dragging
|
||||||
|
, ((modMask, button3), (\w -> focus w >> mouseResizeWindow w))
|
||||||
|
-- you may also bind events to the mouse scroll wheel (button4 and button5)
|
||||||
|
]
|
||||||
|
|
||||||
|
-- | And, finally, the default set of configuration values itself
|
||||||
|
defaultConfig = XConfig
|
||||||
|
{ XMonad.borderWidth = borderWidth
|
||||||
|
, XMonad.workspaces = workspaces
|
||||||
|
, XMonad.defaultGaps = defaultGaps
|
||||||
|
, XMonad.layoutHook = layout
|
||||||
|
, XMonad.terminal = terminal
|
||||||
|
, XMonad.normalBorderColor = normalBorderColor
|
||||||
|
, XMonad.focusedBorderColor = focusedBorderColor
|
||||||
|
, XMonad.numlockMask = numlockMask
|
||||||
|
, XMonad.modMask = defaultModMask
|
||||||
|
, XMonad.keys = keys
|
||||||
|
, XMonad.logHook = logHook
|
||||||
|
, XMonad.mouseBindings = mouseBindings
|
||||||
|
, XMonad.manageHook = manageHook
|
||||||
|
, XMonad.focusFollowsMouse = focusFollowsMouse }
|
391
XMonad/Core.hs
Normal file
391
XMonad/Core.hs
Normal file
@@ -0,0 +1,391 @@
|
|||||||
|
{-# LANGUAGE ExistentialQuantification, FlexibleInstances, GeneralizedNewtypeDeriving,
|
||||||
|
MultiParamTypeClasses, TypeSynonymInstances #-}
|
||||||
|
-- required for deriving Typeable
|
||||||
|
{-# OPTIONS_GHC -fglasgow-exts #-}
|
||||||
|
|
||||||
|
-----------------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : XMonad.Core
|
||||||
|
-- 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.Core (
|
||||||
|
X, WindowSet, WindowSpace, WorkspaceId,
|
||||||
|
ScreenId(..), ScreenDetail(..), XState(..),
|
||||||
|
XConf(..), XConfig(..), LayoutClass(..),
|
||||||
|
Layout(..), readsLayout, Typeable, Message,
|
||||||
|
SomeMessage(..), fromMessage, runLayout, LayoutMessages(..),
|
||||||
|
runX, catchX, userCode, io, catchIO, doubleFork,
|
||||||
|
withDisplay, withWindowSet, isRoot, runOnWorkspaces, broadcastMessage,
|
||||||
|
getAtom, spawn, restart, getXMonadDir, recompile, trace, whenJust, whenX,
|
||||||
|
atom_WM_STATE, atom_WM_PROTOCOLS, atom_WM_DELETE_WINDOW, ManageHook, Query(..), runManageHook
|
||||||
|
) where
|
||||||
|
|
||||||
|
import XMonad.StackSet hiding (modify)
|
||||||
|
|
||||||
|
import Prelude hiding ( catch )
|
||||||
|
import Control.Exception (catch, bracket, throw, Exception(ExitException))
|
||||||
|
import Control.Applicative
|
||||||
|
import Control.Monad.State
|
||||||
|
import Control.Monad.Reader
|
||||||
|
import System.IO
|
||||||
|
import System.Info
|
||||||
|
import System.Posix.Process (executeFile, forkProcess, getProcessStatus, createSession)
|
||||||
|
import System.Process
|
||||||
|
import System.Directory
|
||||||
|
import System.Exit
|
||||||
|
import Graphics.X11.Xlib
|
||||||
|
import Graphics.X11.Xlib.Extras (Event)
|
||||||
|
import Data.Typeable
|
||||||
|
import Data.Monoid
|
||||||
|
|
||||||
|
import qualified Data.Map as M
|
||||||
|
import qualified Data.Set as S
|
||||||
|
|
||||||
|
-- | XState, the window manager state.
|
||||||
|
-- Just the display, width, height and a window list
|
||||||
|
data XState = XState
|
||||||
|
{ windowset :: !WindowSet -- ^ workspace list
|
||||||
|
, mapped :: !(S.Set Window) -- ^ the Set of mapped windows
|
||||||
|
, waitingUnmap :: !(M.Map Window Int) -- ^ the number of expected UnmapEvents
|
||||||
|
, dragging :: !(Maybe (Position -> Position -> X (), X ())) }
|
||||||
|
|
||||||
|
data XConf = XConf
|
||||||
|
{ display :: Display -- ^ the X11 display
|
||||||
|
, config :: !(XConfig Layout) -- ^ initial user configuration
|
||||||
|
, theRoot :: !Window -- ^ the root window
|
||||||
|
, normalBorder :: !Pixel -- ^ border color of unfocused windows
|
||||||
|
, focusedBorder :: !Pixel -- ^ border color of the focused window
|
||||||
|
, keyActions :: !(M.Map (KeyMask, KeySym) (X ()))
|
||||||
|
-- ^ a mapping of key presses to actions
|
||||||
|
, buttonActions :: !(M.Map (KeyMask, Button) (Window -> X ()))
|
||||||
|
-- ^ a mapping of button presses to actions
|
||||||
|
}
|
||||||
|
|
||||||
|
-- todo, better name
|
||||||
|
data XConfig l = XConfig
|
||||||
|
{ normalBorderColor :: !String -- ^ Non focused windows border color. Default: \"#dddddd\"
|
||||||
|
, focusedBorderColor :: !String -- ^ Focused windows border color. Default: \"#ff0000\"
|
||||||
|
, terminal :: !String -- ^ The preferred terminal application. Default: \"xterm\"
|
||||||
|
, layoutHook :: !(l Window) -- ^ The available layouts
|
||||||
|
, manageHook :: !ManageHook -- ^ The action to run when a new window is opened
|
||||||
|
, workspaces :: ![String] -- ^ The list of workspaces' names
|
||||||
|
, defaultGaps :: ![(Int,Int,Int,Int)] -- ^ The list of gaps, per screen
|
||||||
|
, numlockMask :: !KeyMask -- ^ The numlock modifier
|
||||||
|
, modMask :: !KeyMask -- ^ the mod modifier
|
||||||
|
, keys :: !(XConfig Layout -> M.Map (ButtonMask,KeySym) (X ()))
|
||||||
|
-- ^ The key binding: a map from key presses and actions
|
||||||
|
, mouseBindings :: !(XConfig Layout -> M.Map (ButtonMask, Button) (Window -> X ()))
|
||||||
|
-- ^ The mouse bindings
|
||||||
|
, borderWidth :: !Dimension -- ^ The border width
|
||||||
|
, logHook :: !(X ()) -- ^ The action to perform when the windows set is changed
|
||||||
|
, focusFollowsMouse :: !Bool -- ^ Whether window entry events can change focus
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
type WindowSet = StackSet WorkspaceId (Layout Window) Window ScreenId ScreenDetail
|
||||||
|
type WindowSpace = Workspace WorkspaceId (Layout Window) Window
|
||||||
|
|
||||||
|
-- | Virtual workspace indicies
|
||||||
|
type WorkspaceId = String
|
||||||
|
|
||||||
|
-- | Physical screen indicies
|
||||||
|
newtype ScreenId = S Int deriving (Eq,Ord,Show,Read,Enum,Num,Integral,Real)
|
||||||
|
|
||||||
|
-- | The 'Rectangle' with screen dimensions and the list of gaps
|
||||||
|
data ScreenDetail = SD { screenRect :: !Rectangle
|
||||||
|
, statusGap :: !(Int,Int,Int,Int) -- ^ width of status bar on the screen
|
||||||
|
} deriving (Eq,Show, Read)
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- | 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)
|
||||||
|
#ifndef __HADDOCK__
|
||||||
|
deriving (Functor, Monad, MonadIO, MonadState XState, MonadReader XConf)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
instance (Monoid a) => Monoid (X a) where
|
||||||
|
mempty = return mempty
|
||||||
|
mappend = liftM2 mappend
|
||||||
|
|
||||||
|
type ManageHook = Query (Endo WindowSet)
|
||||||
|
newtype Query a = Query (ReaderT Window X a)
|
||||||
|
#ifndef __HADDOCK__
|
||||||
|
deriving (Functor, Monad, MonadReader Window, MonadIO)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
runManageHook :: ManageHook -> Window -> X (WindowSet -> WindowSet)
|
||||||
|
runManageHook (Query m) w = appEndo <$> runReaderT m w
|
||||||
|
|
||||||
|
instance Monoid a => Monoid (Query a) where
|
||||||
|
mempty = return mempty
|
||||||
|
mappend = liftM2 mappend
|
||||||
|
|
||||||
|
-- | 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 (a, XState)
|
||||||
|
runX c st (X a) = runStateT (runReaderT a c) st
|
||||||
|
|
||||||
|
-- | Run in the X monad, and in case of exception, and catch it and log it
|
||||||
|
-- to stderr, and run the error case.
|
||||||
|
catchX :: X a -> X a -> X a
|
||||||
|
catchX job errcase = do
|
||||||
|
st <- get
|
||||||
|
c <- ask
|
||||||
|
(a, s') <- io $ runX c st job `catch` \e -> case e of
|
||||||
|
ExitException {} -> throw e
|
||||||
|
_ -> do hPrint stderr e; runX c st errcase
|
||||||
|
put s'
|
||||||
|
return a
|
||||||
|
|
||||||
|
-- | Execute the argument, catching all exceptions. Either this function or
|
||||||
|
-- catchX should be used at all callsites of user customized code.
|
||||||
|
userCode :: X () -> X ()
|
||||||
|
userCode a = catchX (a >> return ()) (return ())
|
||||||
|
|
||||||
|
-- ---------------------------------------------------------------------
|
||||||
|
-- Convenient wrappers to state
|
||||||
|
|
||||||
|
-- | Run a monad action with the current display settings
|
||||||
|
withDisplay :: (Display -> X a) -> X a
|
||||||
|
withDisplay f = asks display >>= f
|
||||||
|
|
||||||
|
-- | Run a monadic action with the current stack set
|
||||||
|
withWindowSet :: (WindowSet -> X a) -> X a
|
||||||
|
withWindowSet f = gets windowset >>= f
|
||||||
|
|
||||||
|
-- | True if the given window is the root window
|
||||||
|
isRoot :: Window -> X Bool
|
||||||
|
isRoot w = (w==) <$> asks theRoot
|
||||||
|
|
||||||
|
-- | Wrapper for the common case of atom internment
|
||||||
|
getAtom :: String -> X Atom
|
||||||
|
getAtom str = withDisplay $ \dpy -> io $ internAtom dpy str False
|
||||||
|
|
||||||
|
-- | Common non-predefined atoms
|
||||||
|
atom_WM_PROTOCOLS, atom_WM_DELETE_WINDOW, atom_WM_STATE :: X Atom
|
||||||
|
atom_WM_PROTOCOLS = getAtom "WM_PROTOCOLS"
|
||||||
|
atom_WM_DELETE_WINDOW = getAtom "WM_DELETE_WINDOW"
|
||||||
|
atom_WM_STATE = getAtom "WM_STATE"
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- | LayoutClass handling. See particular instances in Operations.hs
|
||||||
|
|
||||||
|
-- | An existential type that can hold any object that is in Read and LayoutClass.
|
||||||
|
data Layout a = forall l. (LayoutClass l a, Read (l a)) => Layout (l a)
|
||||||
|
|
||||||
|
-- | Using the 'Layout' as a witness, parse existentially wrapped windows
|
||||||
|
-- from a 'String'
|
||||||
|
readsLayout :: Layout a -> String -> [(Layout a, String)]
|
||||||
|
readsLayout (Layout l) s = [(Layout (asTypeOf x l), rs) | (x, rs) <- reads s]
|
||||||
|
|
||||||
|
-- | The different layout modes
|
||||||
|
--
|
||||||
|
-- 'doLayout': given a Rectangle and a Stack, layout the stack elements
|
||||||
|
-- inside the given Rectangle. If an element is not given a Rectangle
|
||||||
|
-- by 'doLayout', then it is not shown on screen. Windows are restacked
|
||||||
|
-- according to the order they are returned by 'doLayout'.
|
||||||
|
--
|
||||||
|
class Show (layout a) => LayoutClass layout a where
|
||||||
|
|
||||||
|
-- | Given a Rectangle in which to place the windows, and a Stack of
|
||||||
|
-- windows, return a list of windows and their corresponding Rectangles.
|
||||||
|
-- The order of windows in this list should be the desired stacking order.
|
||||||
|
-- Also return a modified layout, if this layout needs to be modified
|
||||||
|
-- (e.g. if we keep track of the windows we have displayed).
|
||||||
|
doLayout :: layout a -> Rectangle -> Stack a -> X ([(a, Rectangle)], Maybe (layout a))
|
||||||
|
doLayout l r s = return (pureLayout l r s, Nothing)
|
||||||
|
|
||||||
|
-- | This is a pure version of doLayout, for cases where we don't need
|
||||||
|
-- access to the X monad to determine how to layout the windows, and
|
||||||
|
-- we don't need to modify our layout itself.
|
||||||
|
pureLayout :: layout a -> Rectangle -> Stack a -> [(a, Rectangle)]
|
||||||
|
pureLayout _ r s = [(focus s, r)]
|
||||||
|
|
||||||
|
-- | 'handleMessage' performs message handling for that layout. If
|
||||||
|
-- 'handleMessage' returns Nothing, then the layout did not respond to
|
||||||
|
-- that message and the screen is not refreshed. Otherwise, 'handleMessage'
|
||||||
|
-- returns an updated 'Layout' and the screen is refreshed.
|
||||||
|
--
|
||||||
|
handleMessage :: layout a -> SomeMessage -> X (Maybe (layout a))
|
||||||
|
handleMessage l = return . pureMessage l
|
||||||
|
|
||||||
|
-- | Respond to a message by (possibly) changing our layout, but taking
|
||||||
|
-- no other action. If the layout changes, the screen will be refreshed.
|
||||||
|
pureMessage :: layout a -> SomeMessage -> Maybe (layout a)
|
||||||
|
pureMessage _ _ = Nothing
|
||||||
|
|
||||||
|
-- | This should be a human-readable string that is used when selecting
|
||||||
|
-- layouts by name.
|
||||||
|
description :: layout a -> String
|
||||||
|
description = show
|
||||||
|
|
||||||
|
instance LayoutClass Layout Window where
|
||||||
|
doLayout (Layout l) r s = fmap (fmap Layout) `fmap` doLayout l r s
|
||||||
|
handleMessage (Layout l) = fmap (fmap Layout) . handleMessage l
|
||||||
|
description (Layout l) = description l
|
||||||
|
|
||||||
|
instance Show (Layout a) where show (Layout l) = show l
|
||||||
|
|
||||||
|
-- | This calls doLayout if there are any windows to be laid out.
|
||||||
|
runLayout :: LayoutClass l a => l a -> Rectangle -> Maybe (Stack a) -> X ([(a, Rectangle)], Maybe (l a))
|
||||||
|
runLayout l r = maybe (return ([], Nothing)) (doLayout l r)
|
||||||
|
|
||||||
|
-- | Based on ideas in /An Extensible Dynamically-Typed Hierarchy of Exceptions/,
|
||||||
|
-- Simon Marlow, 2006. Use extensible messages to the handleMessage 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
|
||||||
|
|
||||||
|
-- | X Events are valid Messages
|
||||||
|
instance Message Event
|
||||||
|
|
||||||
|
-- | LayoutMessages are core messages that all layouts (especially stateful
|
||||||
|
-- layouts) should consider handling.
|
||||||
|
data LayoutMessages = Hide -- ^ sent when a layout becomes non-visible
|
||||||
|
| ReleaseResources -- ^ sent when xmonad is exiting or restarting
|
||||||
|
deriving (Typeable, Eq)
|
||||||
|
|
||||||
|
instance Message LayoutMessages
|
||||||
|
|
||||||
|
-- ---------------------------------------------------------------------
|
||||||
|
-- | General utilities
|
||||||
|
--
|
||||||
|
-- Lift an IO action into the X monad
|
||||||
|
io :: MonadIO m => IO a -> m a
|
||||||
|
io = liftIO
|
||||||
|
|
||||||
|
-- | Lift an IO action into the X monad. If the action results in an IO
|
||||||
|
-- exception, log the exception to stderr and continue normal execution.
|
||||||
|
catchIO :: IO () -> X ()
|
||||||
|
catchIO f = io (f `catch` \e -> hPrint stderr e >> hFlush stderr)
|
||||||
|
|
||||||
|
-- | spawn. Launch an external application
|
||||||
|
spawn :: MonadIO m => String -> m ()
|
||||||
|
spawn x = doubleFork $ executeFile "/bin/sh" False ["-c", x] Nothing
|
||||||
|
|
||||||
|
-- | Double fork and execute an IO action (usually one of the exec family of
|
||||||
|
-- functions)
|
||||||
|
doubleFork :: MonadIO m => IO () -> m ()
|
||||||
|
doubleFork m = io $ do
|
||||||
|
pid <- forkProcess $ do
|
||||||
|
forkProcess (createSession >> m)
|
||||||
|
exitWith ExitSuccess
|
||||||
|
getProcessStatus True False pid
|
||||||
|
return ()
|
||||||
|
|
||||||
|
-- | Send a message to all visible layouts, without necessarily refreshing.
|
||||||
|
-- This is how we implement the hooks, such as UnDoLayout.
|
||||||
|
broadcastMessage :: Message a => a -> X ()
|
||||||
|
broadcastMessage a = runOnWorkspaces $ \w -> do
|
||||||
|
ml' <- handleMessage (layout w) (SomeMessage a) `catchX` return Nothing
|
||||||
|
return $ w { layout = maybe (layout w) id ml' }
|
||||||
|
|
||||||
|
-- | This is basically a map function, running a function in the X monad on
|
||||||
|
-- each workspace with the output of that function being the modified workspace.
|
||||||
|
runOnWorkspaces :: (WindowSpace -> X WindowSpace) -> X ()
|
||||||
|
runOnWorkspaces job = do
|
||||||
|
ws <- gets windowset
|
||||||
|
h <- mapM job $ hidden ws
|
||||||
|
c:v <- mapM (\s -> (\w -> s { workspace = w}) <$> job (workspace s))
|
||||||
|
$ current ws : visible ws
|
||||||
|
modify $ \s -> s { windowset = ws { current = c, visible = v, hidden = h } }
|
||||||
|
|
||||||
|
-- | @restart name resume@. Attempt to restart xmonad by executing the program
|
||||||
|
-- @name@. If @resume@ is 'True', restart with the current window state.
|
||||||
|
-- When executing another window manager, @resume@ should be 'False'.
|
||||||
|
--
|
||||||
|
restart :: String -> Bool -> X ()
|
||||||
|
restart prog resume = do
|
||||||
|
broadcastMessage ReleaseResources
|
||||||
|
io . flush =<< asks display
|
||||||
|
args <- if resume then gets (("--resume":) . return . showWs . windowset) else return []
|
||||||
|
catchIO (executeFile prog True args Nothing)
|
||||||
|
where showWs = show . mapLayout show
|
||||||
|
|
||||||
|
-- | Return the path to @~\/.xmonad@.
|
||||||
|
getXMonadDir :: MonadIO m => m String
|
||||||
|
getXMonadDir = io $ getAppUserDataDirectory "xmonad"
|
||||||
|
|
||||||
|
-- | 'recompile force', recompile ~\/.xmonad\/xmonad.hs when any of the
|
||||||
|
-- following apply:
|
||||||
|
-- * force is True
|
||||||
|
-- * the xmonad executable does not exist
|
||||||
|
-- * the xmonad executable is older than xmonad.hs
|
||||||
|
--
|
||||||
|
-- The -i flag is used to restrict recompilation to the xmonad.hs file only.
|
||||||
|
--
|
||||||
|
-- Compilation errors (if any) are logged to ~\/.xmonad\/xmonad.errors. If
|
||||||
|
-- GHC indicates failure with a non-zero exit code, an xmessage displaying
|
||||||
|
-- that file is spawned.
|
||||||
|
--
|
||||||
|
-- False is returned if there is compilation errors.
|
||||||
|
--
|
||||||
|
recompile :: MonadIO m => Bool -> m Bool
|
||||||
|
recompile force = io $ do
|
||||||
|
dir <- getXMonadDir
|
||||||
|
let binn = "xmonad-"++arch++"-"++os
|
||||||
|
bin = dir ++ "/" ++ binn
|
||||||
|
base = dir ++ "/" ++ "xmonad"
|
||||||
|
err = base ++ ".errors"
|
||||||
|
src = base ++ ".hs"
|
||||||
|
srcT <- getModTime src
|
||||||
|
binT <- getModTime bin
|
||||||
|
if (force || srcT > binT)
|
||||||
|
then do
|
||||||
|
status <- bracket (openFile err WriteMode) hClose $ \h -> do
|
||||||
|
waitForProcess =<< runProcess "ghc" ["--make", "xmonad.hs", "-i", "-no-recomp", "-v0", "-o",binn] (Just dir)
|
||||||
|
Nothing Nothing Nothing (Just h)
|
||||||
|
|
||||||
|
-- now, if it fails, run xmessage to let the user know:
|
||||||
|
when (status /= ExitSuccess) $ do
|
||||||
|
ghcErr <- readFile err
|
||||||
|
let msg = unlines $
|
||||||
|
["Error detected while loading xmonad configuration file: " ++ src]
|
||||||
|
++ lines ghcErr ++ ["","Please check the file for errors."]
|
||||||
|
doubleFork $ executeFile "xmessage" True [msg] Nothing
|
||||||
|
return (status == ExitSuccess)
|
||||||
|
else return True
|
||||||
|
where getModTime f = catch (Just <$> getModificationTime f) (const $ return Nothing)
|
||||||
|
|
||||||
|
-- | Run a side effecting action with the current workspace. Like 'when' but
|
||||||
|
whenJust :: Monad m => Maybe a -> (a -> m ()) -> m ()
|
||||||
|
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
|
||||||
|
|
||||||
|
-- | A 'trace' for the X monad. Logs a string to stderr. The result may
|
||||||
|
-- be found in your .xsession-errors file
|
||||||
|
trace :: MonadIO m => String -> m ()
|
||||||
|
trace = io . hPutStrLn stderr
|
175
XMonad/Layout.hs
Normal file
175
XMonad/Layout.hs
Normal file
@@ -0,0 +1,175 @@
|
|||||||
|
{-# OPTIONS_GHC -fglasgow-exts #-} -- For deriving Data/Typeable
|
||||||
|
{-# LANGUAGE FlexibleInstances, MultiParamTypeClasses, PatternGuards, TypeSynonymInstances #-}
|
||||||
|
|
||||||
|
-- --------------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : XMonad.Layout
|
||||||
|
-- Copyright : (c) Spencer Janssen 2007
|
||||||
|
-- License : BSD3-style (see LICENSE)
|
||||||
|
--
|
||||||
|
-- Maintainer : sjanssen@cse.unl.edu
|
||||||
|
-- Stability : unstable
|
||||||
|
-- Portability : not portable, Typeable deriving, mtl, posix
|
||||||
|
--
|
||||||
|
-- The collection of core layouts.
|
||||||
|
--
|
||||||
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
module XMonad.Layout (ChangeLayout(..), Choose, (|||), Resize(..), IncMasterN(..),
|
||||||
|
Full(..), Tall(..), Mirror(..), mirrorRect, splitVertically,
|
||||||
|
splitHorizontally, splitHorizontallyBy, splitVerticallyBy) where
|
||||||
|
|
||||||
|
import XMonad.Core
|
||||||
|
|
||||||
|
import Graphics.X11 (Rectangle(..))
|
||||||
|
import qualified XMonad.StackSet as W
|
||||||
|
import Control.Arrow ((***), second)
|
||||||
|
import Control.Monad
|
||||||
|
import Data.Maybe (fromMaybe)
|
||||||
|
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- LayoutClass selection manager
|
||||||
|
|
||||||
|
-- | A layout that allows users to switch between various layout options.
|
||||||
|
|
||||||
|
-- | Messages to change the current layout.
|
||||||
|
data ChangeLayout = FirstLayout | NextLayout deriving (Eq, Show, Typeable)
|
||||||
|
|
||||||
|
instance Message ChangeLayout
|
||||||
|
|
||||||
|
-- | The layout choice combinator
|
||||||
|
(|||) :: (LayoutClass l a, LayoutClass r a) => l a -> r a -> Choose l r a
|
||||||
|
(|||) = flip SLeft
|
||||||
|
infixr 5 |||
|
||||||
|
|
||||||
|
data Choose l r a = SLeft (r a) (l a)
|
||||||
|
| SRight (l a) (r a) deriving (Read, Show)
|
||||||
|
|
||||||
|
data NextNoWrap = NextNoWrap deriving (Eq, Show, Typeable)
|
||||||
|
instance Message NextNoWrap
|
||||||
|
|
||||||
|
-- This has lots of pseudo duplicated code, we must find a better way
|
||||||
|
instance (LayoutClass l a, LayoutClass r a) => LayoutClass (Choose l r) a where
|
||||||
|
doLayout (SLeft r l) = (fmap (second . fmap $ SLeft r) .) . doLayout l
|
||||||
|
doLayout (SRight l r) = (fmap (second . fmap $ SRight l) .) . doLayout r
|
||||||
|
|
||||||
|
description (SLeft _ l) = description l
|
||||||
|
description (SRight _ r) = description r
|
||||||
|
|
||||||
|
handleMessage lr m | Just FirstLayout <- fromMessage m = case lr of
|
||||||
|
SLeft {} -> return Nothing
|
||||||
|
SRight l r -> fmap (Just . flip SLeft l . fromMaybe r)
|
||||||
|
$ handleMessage r (SomeMessage Hide)
|
||||||
|
|
||||||
|
handleMessage lr m | Just NextLayout <- fromMessage m = do
|
||||||
|
mlr <- handleMessage lr $ SomeMessage NextNoWrap
|
||||||
|
maybe (handleMessage lr $ SomeMessage FirstLayout) (return . Just) mlr
|
||||||
|
|
||||||
|
handleMessage (SLeft r l) m | Just NextNoWrap <- fromMessage m = do
|
||||||
|
handleMessage l (SomeMessage Hide)
|
||||||
|
mr <- handleMessage r (SomeMessage FirstLayout)
|
||||||
|
return . Just . SRight l $ fromMaybe r mr
|
||||||
|
|
||||||
|
handleMessage lr m | Just ReleaseResources <- fromMessage m =
|
||||||
|
liftM2 ((Just .) . cons)
|
||||||
|
(fmap (fromMaybe l) $ handleMessage l m)
|
||||||
|
(fmap (fromMaybe r) $ handleMessage r m)
|
||||||
|
where (cons, l, r) = case lr of
|
||||||
|
(SLeft r' l') -> (flip SLeft, l', r')
|
||||||
|
(SRight l' r') -> (SRight, l', r')
|
||||||
|
|
||||||
|
-- The default cases for left and right:
|
||||||
|
handleMessage (SLeft r l) m = fmap (fmap $ SLeft r) $ handleMessage l m
|
||||||
|
handleMessage (SRight l r) m = fmap (fmap $ SRight l) $ handleMessage r m
|
||||||
|
|
||||||
|
--
|
||||||
|
-- | Builtin layout algorithms:
|
||||||
|
--
|
||||||
|
-- > fullscreen mode
|
||||||
|
-- > tall mode
|
||||||
|
--
|
||||||
|
-- The latter algorithms support the following operations:
|
||||||
|
--
|
||||||
|
-- > Shrink
|
||||||
|
-- > Expand
|
||||||
|
--
|
||||||
|
data Resize = Shrink | Expand deriving Typeable
|
||||||
|
|
||||||
|
-- | You can also increase the number of clients in the master pane
|
||||||
|
data IncMasterN = IncMasterN Int deriving Typeable
|
||||||
|
|
||||||
|
instance Message Resize
|
||||||
|
instance Message IncMasterN
|
||||||
|
|
||||||
|
-- | Simple fullscreen mode, just render all windows fullscreen.
|
||||||
|
data Full a = Full deriving (Show, Read)
|
||||||
|
|
||||||
|
instance LayoutClass Full a
|
||||||
|
|
||||||
|
-- | The inbuilt tiling mode of xmonad, and its operations.
|
||||||
|
data Tall a = Tall Int Rational Rational deriving (Show, Read)
|
||||||
|
|
||||||
|
instance LayoutClass Tall a where
|
||||||
|
pureLayout (Tall nmaster _ frac) r s = zip ws rs
|
||||||
|
where ws = W.integrate s
|
||||||
|
rs = tile frac r nmaster (length ws)
|
||||||
|
|
||||||
|
pureMessage (Tall nmaster delta frac) m = msum [fmap resize (fromMessage m)
|
||||||
|
,fmap incmastern (fromMessage m)]
|
||||||
|
where resize Shrink = Tall nmaster delta (max 0 $ frac-delta)
|
||||||
|
resize Expand = Tall nmaster delta (min 1 $ frac+delta)
|
||||||
|
incmastern (IncMasterN d) = Tall (max 0 (nmaster+d)) delta frac
|
||||||
|
description _ = "Tall"
|
||||||
|
|
||||||
|
-- | Mirror a rectangle
|
||||||
|
mirrorRect :: Rectangle -> Rectangle
|
||||||
|
mirrorRect (Rectangle rx ry rw rh) = (Rectangle ry rx rh rw)
|
||||||
|
|
||||||
|
-- | Mirror a layout, compute its 90 degree rotated form.
|
||||||
|
data Mirror l a = Mirror (l a) deriving (Show, Read)
|
||||||
|
|
||||||
|
instance LayoutClass l a => LayoutClass (Mirror l) a where
|
||||||
|
doLayout (Mirror l) r s = (map (second mirrorRect) *** fmap Mirror)
|
||||||
|
`fmap` doLayout l (mirrorRect r) s
|
||||||
|
handleMessage (Mirror l) = fmap (fmap Mirror) . handleMessage l
|
||||||
|
description (Mirror l) = "Mirror "++ description l
|
||||||
|
|
||||||
|
-- | tile. Compute the positions for windows using the default 2 pane tiling algorithm.
|
||||||
|
--
|
||||||
|
-- The screen is divided (currently) into two panes. all clients are
|
||||||
|
-- then partioned between these two panes. one pane, the `master', by
|
||||||
|
-- convention has the least number of windows in it (by default, 1).
|
||||||
|
-- the variable `nmaster' controls how many windows are rendered in the
|
||||||
|
-- master pane.
|
||||||
|
--
|
||||||
|
-- `delta' specifies the ratio of the screen to resize by.
|
||||||
|
--
|
||||||
|
-- 'frac' specifies what proportion of the screen to devote to the
|
||||||
|
-- master area.
|
||||||
|
--
|
||||||
|
tile :: Rational -> Rectangle -> Int -> Int -> [Rectangle]
|
||||||
|
tile f r nmaster n = if n <= nmaster || nmaster == 0
|
||||||
|
then splitVertically n r
|
||||||
|
else splitVertically nmaster r1 ++ splitVertically (n-nmaster) r2 -- two columns
|
||||||
|
where (r1,r2) = splitHorizontallyBy f r
|
||||||
|
|
||||||
|
--
|
||||||
|
-- Divide the screen vertically into n subrectangles
|
||||||
|
--
|
||||||
|
splitVertically, splitHorizontally :: Int -> Rectangle -> [Rectangle]
|
||||||
|
splitVertically n r | n < 2 = [r]
|
||||||
|
splitVertically n (Rectangle sx sy sw sh) = Rectangle sx sy sw smallh :
|
||||||
|
splitVertically (n-1) (Rectangle sx (sy+fromIntegral smallh) sw (sh-smallh))
|
||||||
|
where smallh = sh `div` fromIntegral n --hmm, this is a fold or map.
|
||||||
|
|
||||||
|
splitHorizontally n = map mirrorRect . splitVertically n . mirrorRect
|
||||||
|
|
||||||
|
-- Divide the screen into two rectangles, using a rational to specify the ratio
|
||||||
|
splitHorizontallyBy, splitVerticallyBy :: RealFrac r => r -> Rectangle -> (Rectangle, Rectangle)
|
||||||
|
splitHorizontallyBy f (Rectangle sx sy sw sh) =
|
||||||
|
( Rectangle sx sy leftw sh
|
||||||
|
, Rectangle (sx + fromIntegral leftw) sy (sw-fromIntegral leftw) sh)
|
||||||
|
where leftw = floor $ fromIntegral sw * f
|
||||||
|
|
||||||
|
splitVerticallyBy f = (mirrorRect *** mirrorRect) . splitHorizontallyBy f . mirrorRect
|
284
XMonad/Main.hs
Normal file
284
XMonad/Main.hs
Normal file
@@ -0,0 +1,284 @@
|
|||||||
|
{-# LANGUAGE MultiParamTypeClasses, FlexibleContexts #-}
|
||||||
|
----------------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : XMonad.Main
|
||||||
|
-- Copyright : (c) Spencer Janssen 2007
|
||||||
|
-- License : BSD3-style (see LICENSE)
|
||||||
|
--
|
||||||
|
-- Maintainer : sjanssen@cse.unl.edu
|
||||||
|
-- Stability : unstable
|
||||||
|
-- Portability : not portable, uses mtl, X11, posix
|
||||||
|
--
|
||||||
|
-- xmonad, a minimalist, tiling window manager for X11
|
||||||
|
--
|
||||||
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
module XMonad.Main (xmonad) where
|
||||||
|
|
||||||
|
import Data.Bits
|
||||||
|
import qualified Data.Map as M
|
||||||
|
import qualified Data.Set as S
|
||||||
|
import Control.Monad.Reader
|
||||||
|
import Control.Monad.State
|
||||||
|
import Data.Maybe (fromMaybe)
|
||||||
|
|
||||||
|
import System.Environment (getArgs)
|
||||||
|
|
||||||
|
import Graphics.X11.Xlib hiding (refreshKeyboardMapping)
|
||||||
|
import Graphics.X11.Xlib.Extras
|
||||||
|
|
||||||
|
import XMonad.Core
|
||||||
|
import XMonad.StackSet (new, floating, member)
|
||||||
|
import qualified XMonad.StackSet as W
|
||||||
|
import XMonad.Operations
|
||||||
|
|
||||||
|
import System.IO
|
||||||
|
|
||||||
|
-- |
|
||||||
|
-- The main entry point
|
||||||
|
--
|
||||||
|
xmonad :: (LayoutClass l Window, Read (l Window)) => XConfig l -> IO ()
|
||||||
|
xmonad initxmc = do
|
||||||
|
-- First, wrap the layout in an existential, to keep things pretty:
|
||||||
|
let xmc = initxmc { layoutHook = Layout $ layoutHook initxmc }
|
||||||
|
dpy <- openDisplay ""
|
||||||
|
let dflt = defaultScreen dpy
|
||||||
|
|
||||||
|
rootw <- rootWindow dpy dflt
|
||||||
|
xinesc <- getCleanedScreenInfo dpy
|
||||||
|
nbc <- initColor dpy $ normalBorderColor xmc
|
||||||
|
fbc <- initColor dpy $ focusedBorderColor xmc
|
||||||
|
hSetBuffering stdout NoBuffering
|
||||||
|
args <- getArgs
|
||||||
|
|
||||||
|
let layout = layoutHook xmc
|
||||||
|
lreads = readsLayout layout
|
||||||
|
initialWinset = new layout (workspaces xmc) $ zipWith SD xinesc gaps
|
||||||
|
|
||||||
|
maybeRead reads' s = case reads' s of
|
||||||
|
[(x, "")] -> Just x
|
||||||
|
_ -> Nothing
|
||||||
|
|
||||||
|
winset = fromMaybe initialWinset $ do
|
||||||
|
("--resume" : s : _) <- return args
|
||||||
|
ws <- maybeRead reads s
|
||||||
|
return . W.ensureTags layout (workspaces xmc)
|
||||||
|
$ W.mapLayout (fromMaybe layout . maybeRead lreads) ws
|
||||||
|
|
||||||
|
gaps = take (length xinesc) $ defaultGaps xmc ++ repeat (0,0,0,0)
|
||||||
|
|
||||||
|
cf = XConf
|
||||||
|
{ display = dpy
|
||||||
|
, config = xmc
|
||||||
|
, theRoot = rootw
|
||||||
|
, normalBorder = nbc
|
||||||
|
, focusedBorder = fbc
|
||||||
|
, keyActions = keys xmc xmc
|
||||||
|
, buttonActions = mouseBindings xmc xmc }
|
||||||
|
st = XState
|
||||||
|
{ windowset = initialWinset
|
||||||
|
, mapped = S.empty
|
||||||
|
, waitingUnmap = M.empty
|
||||||
|
, dragging = Nothing }
|
||||||
|
|
||||||
|
xSetErrorHandler -- in C, I'm too lazy to write the binding: dons
|
||||||
|
|
||||||
|
-- setup initial X environment
|
||||||
|
sync dpy False
|
||||||
|
selectInput dpy rootw $ substructureRedirectMask .|. substructureNotifyMask
|
||||||
|
.|. enterWindowMask .|. leaveWindowMask .|. structureNotifyMask
|
||||||
|
|
||||||
|
allocaXEvent $ \e ->
|
||||||
|
runX cf st $ do
|
||||||
|
|
||||||
|
grabKeys
|
||||||
|
grabButtons
|
||||||
|
|
||||||
|
io $ sync dpy False
|
||||||
|
|
||||||
|
-- bootstrap the windowset, Operations.windows will identify all
|
||||||
|
-- the windows in winset as new and set initial properties for
|
||||||
|
-- those windows
|
||||||
|
windows (const winset)
|
||||||
|
|
||||||
|
-- scan for all top-level windows, add the unmanaged ones to the
|
||||||
|
-- windowset
|
||||||
|
ws <- io $ scan dpy rootw
|
||||||
|
mapM_ manage ws
|
||||||
|
|
||||||
|
-- main loop, for all you HOF/recursion fans out there.
|
||||||
|
forever_ $ handle =<< io (nextEvent dpy e >> getEvent e)
|
||||||
|
|
||||||
|
return ()
|
||||||
|
where forever_ a = a >> forever_ a
|
||||||
|
|
||||||
|
|
||||||
|
-- ---------------------------------------------------------------------
|
||||||
|
-- | Event handler. Map X events onto calls into Operations.hs, which
|
||||||
|
-- modify our internal model of the window manager state.
|
||||||
|
--
|
||||||
|
-- Events dwm handles that we don't:
|
||||||
|
--
|
||||||
|
-- [ButtonPress] = buttonpress,
|
||||||
|
-- [Expose] = expose,
|
||||||
|
-- [PropertyNotify] = propertynotify,
|
||||||
|
--
|
||||||
|
handle :: Event -> X ()
|
||||||
|
|
||||||
|
-- run window manager command
|
||||||
|
handle (KeyEvent {ev_event_type = t, ev_state = m, ev_keycode = code})
|
||||||
|
| t == keyPress = withDisplay $ \dpy -> do
|
||||||
|
s <- io $ keycodeToKeysym dpy code 0
|
||||||
|
mClean <- cleanMask m
|
||||||
|
ks <- asks keyActions
|
||||||
|
userCode $ whenJust (M.lookup (mClean, s) ks) id
|
||||||
|
|
||||||
|
-- manage a new window
|
||||||
|
handle (MapRequestEvent {ev_window = w}) = withDisplay $ \dpy -> do
|
||||||
|
wa <- io $ getWindowAttributes dpy w -- ignore override windows
|
||||||
|
-- need to ignore mapping requests by managed windows not on the current workspace
|
||||||
|
managed <- isClient w
|
||||||
|
when (not (wa_override_redirect wa) && not managed) $ do manage w
|
||||||
|
|
||||||
|
-- window destroyed, unmanage it
|
||||||
|
-- window gone, unmanage it
|
||||||
|
handle (DestroyWindowEvent {ev_window = w}) = whenX (isClient w) $ do
|
||||||
|
unmanage w
|
||||||
|
modify (\s -> s { mapped = S.delete w (mapped s)
|
||||||
|
, waitingUnmap = M.delete w (waitingUnmap s)})
|
||||||
|
|
||||||
|
-- We track expected unmap events in waitingUnmap. We ignore this event unless
|
||||||
|
-- it is synthetic or we are not expecting an unmap notification from a window.
|
||||||
|
handle (UnmapEvent {ev_window = w, ev_send_event = synthetic}) = whenX (isClient w) $ do
|
||||||
|
e <- gets (fromMaybe 0 . M.lookup w . waitingUnmap)
|
||||||
|
if (synthetic || e == 0)
|
||||||
|
then unmanage w
|
||||||
|
else modify (\s -> s { waitingUnmap = M.update mpred w (waitingUnmap s) })
|
||||||
|
where mpred 1 = Nothing
|
||||||
|
mpred n = Just $ pred n
|
||||||
|
|
||||||
|
-- set keyboard mapping
|
||||||
|
handle e@(MappingNotifyEvent {}) = do
|
||||||
|
io $ refreshKeyboardMapping e
|
||||||
|
when (ev_request e == mappingKeyboard) grabKeys
|
||||||
|
|
||||||
|
-- handle button release, which may finish dragging.
|
||||||
|
handle e@(ButtonEvent {ev_event_type = t})
|
||||||
|
| t == buttonRelease = do
|
||||||
|
drag <- gets dragging
|
||||||
|
case drag of
|
||||||
|
-- we're done dragging and have released the mouse:
|
||||||
|
Just (_,f) -> modify (\s -> s { dragging = Nothing }) >> f
|
||||||
|
Nothing -> broadcastMessage e
|
||||||
|
|
||||||
|
-- handle motionNotify event, which may mean we are dragging.
|
||||||
|
handle e@(MotionEvent {ev_event_type = _t, ev_x = x, ev_y = y}) = do
|
||||||
|
drag <- gets dragging
|
||||||
|
case drag of
|
||||||
|
Just (d,_) -> d (fromIntegral x) (fromIntegral y) -- we're dragging
|
||||||
|
Nothing -> broadcastMessage e
|
||||||
|
|
||||||
|
-- click on an unfocused window, makes it focused on this workspace
|
||||||
|
handle e@(ButtonEvent {ev_window = w,ev_event_type = t,ev_button = b })
|
||||||
|
| t == buttonPress = do
|
||||||
|
-- If it's the root window, then it's something we
|
||||||
|
-- grabbed in grabButtons. Otherwise, it's click-to-focus.
|
||||||
|
isr <- isRoot w
|
||||||
|
m <- cleanMask $ ev_state e
|
||||||
|
ba <- asks buttonActions
|
||||||
|
if isr then userCode $ whenJust (M.lookup (m, b) ba) ($ ev_subwindow e)
|
||||||
|
else focus w
|
||||||
|
broadcastMessage e -- Always send button events.
|
||||||
|
|
||||||
|
-- entered a normal window, makes this focused.
|
||||||
|
handle e@(CrossingEvent {ev_window = w, ev_event_type = t})
|
||||||
|
| t == enterNotify && ev_mode e == notifyNormal
|
||||||
|
&& ev_detail e /= notifyInferior
|
||||||
|
= whenX (asks $ focusFollowsMouse . config) (focus w)
|
||||||
|
|
||||||
|
-- left a window, check if we need to focus root
|
||||||
|
handle e@(CrossingEvent {ev_event_type = t})
|
||||||
|
| t == leaveNotify
|
||||||
|
= do rootw <- asks theRoot
|
||||||
|
when (ev_window e == rootw && not (ev_same_screen e)) $ setFocusX rootw
|
||||||
|
|
||||||
|
-- configure a window
|
||||||
|
handle e@(ConfigureRequestEvent {ev_window = w}) = withDisplay $ \dpy -> do
|
||||||
|
ws <- gets windowset
|
||||||
|
wa <- io $ getWindowAttributes dpy w
|
||||||
|
|
||||||
|
bw <- asks (borderWidth . config)
|
||||||
|
|
||||||
|
if M.member w (floating ws)
|
||||||
|
|| not (member w ws)
|
||||||
|
then do io $ configureWindow dpy w (ev_value_mask e) $ WindowChanges
|
||||||
|
{ wc_x = ev_x e
|
||||||
|
, wc_y = ev_y e
|
||||||
|
, wc_width = ev_width e
|
||||||
|
, wc_height = ev_height e
|
||||||
|
, wc_border_width = fromIntegral bw
|
||||||
|
, wc_sibling = ev_above e
|
||||||
|
, wc_stack_mode = ev_detail e }
|
||||||
|
when (member w ws) (float w)
|
||||||
|
else io $ allocaXEvent $ \ev -> do
|
||||||
|
setEventType ev configureNotify
|
||||||
|
setConfigureEvent ev w w
|
||||||
|
(wa_x wa) (wa_y wa) (wa_width wa)
|
||||||
|
(wa_height wa) (ev_border_width e) none (wa_override_redirect wa)
|
||||||
|
sendEvent dpy w False 0 ev
|
||||||
|
io $ sync dpy False
|
||||||
|
|
||||||
|
-- configuration changes in the root may mean display settings have changed
|
||||||
|
handle (ConfigureEvent {ev_window = w}) = whenX (isRoot w) rescreen
|
||||||
|
|
||||||
|
-- property notify
|
||||||
|
handle PropertyEvent { ev_event_type = t, ev_atom = a }
|
||||||
|
| t == propertyNotify && a == wM_NAME = userCode =<< asks (logHook . config)
|
||||||
|
|
||||||
|
handle e = broadcastMessage e -- trace (eventName e) -- ignoring
|
||||||
|
|
||||||
|
|
||||||
|
-- ---------------------------------------------------------------------
|
||||||
|
-- IO stuff. Doesn't require any X state
|
||||||
|
-- Most of these things run only on startup (bar grabkeys)
|
||||||
|
|
||||||
|
-- | scan for any new windows to manage. If they're already managed,
|
||||||
|
-- this should be idempotent.
|
||||||
|
scan :: Display -> Window -> IO [Window]
|
||||||
|
scan dpy rootw = do
|
||||||
|
(_, _, ws) <- queryTree dpy rootw
|
||||||
|
filterM ok ws
|
||||||
|
-- TODO: scan for windows that are either 'IsViewable' or where WM_STATE ==
|
||||||
|
-- Iconic
|
||||||
|
where ok w = do wa <- getWindowAttributes dpy w
|
||||||
|
a <- internAtom dpy "WM_STATE" False
|
||||||
|
p <- getWindowProperty32 dpy a w
|
||||||
|
let ic = case p of
|
||||||
|
Just (3:_) -> True -- 3 for iconified
|
||||||
|
_ -> False
|
||||||
|
return $ not (wa_override_redirect wa)
|
||||||
|
&& (wa_map_state wa == waIsViewable || ic)
|
||||||
|
|
||||||
|
-- | Grab the keys back
|
||||||
|
grabKeys :: X ()
|
||||||
|
grabKeys = do
|
||||||
|
XConf { display = dpy, theRoot = rootw } <- ask
|
||||||
|
let grab kc m = io $ grabKey dpy kc m rootw True grabModeAsync grabModeAsync
|
||||||
|
io $ ungrabKey dpy anyKey anyModifier rootw
|
||||||
|
ks <- asks keyActions
|
||||||
|
forM_ (M.keys ks) $ \(mask,sym) -> do
|
||||||
|
kc <- io $ keysymToKeycode dpy sym
|
||||||
|
-- "If the specified KeySym is not defined for any KeyCode,
|
||||||
|
-- XKeysymToKeycode() returns zero."
|
||||||
|
when (kc /= '\0') $ mapM_ (grab kc . (mask .|.)) =<< extraModifiers
|
||||||
|
|
||||||
|
-- | XXX comment me
|
||||||
|
grabButtons :: X ()
|
||||||
|
grabButtons = do
|
||||||
|
XConf { display = dpy, theRoot = rootw } <- ask
|
||||||
|
let grab button mask = io $ grabButton dpy button mask rootw False buttonPressMask
|
||||||
|
grabModeAsync grabModeSync none none
|
||||||
|
io $ ungrabButton dpy anyButton anyModifier rootw
|
||||||
|
ems <- extraModifiers
|
||||||
|
ba <- asks buttonActions
|
||||||
|
mapM_ (\(m,b) -> mapM_ (grab b . (m .|.)) ems) (M.keys $ ba)
|
78
XMonad/ManageHook.hs
Normal file
78
XMonad/ManageHook.hs
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
|
||||||
|
|
||||||
|
-----------------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : XMonad.ManageHook
|
||||||
|
-- Copyright : (c) Spencer Janssen 2007
|
||||||
|
-- License : BSD3-style (see LICENSE)
|
||||||
|
--
|
||||||
|
-- Maintainer : sjanssen@cse.unl.edu
|
||||||
|
-- Stability : unstable
|
||||||
|
-- Portability : not portable, uses cunning newtype deriving
|
||||||
|
--
|
||||||
|
-- An EDSL for ManageHooks
|
||||||
|
--
|
||||||
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- XXX examples required
|
||||||
|
|
||||||
|
module XMonad.ManageHook where
|
||||||
|
|
||||||
|
import XMonad.Core
|
||||||
|
import Graphics.X11.Xlib.Extras
|
||||||
|
import Control.Monad.Reader
|
||||||
|
import Data.Maybe
|
||||||
|
import Data.Monoid
|
||||||
|
import qualified XMonad.StackSet as W
|
||||||
|
import XMonad.Operations (floatLocation, reveal)
|
||||||
|
|
||||||
|
liftX :: X a -> Query a
|
||||||
|
liftX = Query . lift
|
||||||
|
|
||||||
|
-- | The identity hook that returns the WindowSet unchanged.
|
||||||
|
idHook :: ManageHook
|
||||||
|
idHook = doF id
|
||||||
|
|
||||||
|
-- | Compose two 'ManageHook's
|
||||||
|
(<+>) :: ManageHook -> ManageHook -> ManageHook
|
||||||
|
(<+>) = mappend
|
||||||
|
|
||||||
|
-- | Compose the list of 'ManageHook's
|
||||||
|
composeAll :: [ManageHook] -> ManageHook
|
||||||
|
composeAll = mconcat
|
||||||
|
|
||||||
|
-- | 'p --> x'. If 'p' returns 'True', execute the 'ManageHook'.
|
||||||
|
(-->) :: Query Bool -> ManageHook -> ManageHook
|
||||||
|
p --> f = p >>= \b -> if b then f else mempty
|
||||||
|
|
||||||
|
-- | 'q =? x'. if the result of 'q' equals 'x', return 'True'.
|
||||||
|
(=?) :: Eq a => Query a -> a -> Query Bool
|
||||||
|
q =? x = fmap (== x) q
|
||||||
|
|
||||||
|
infixr 3 <&&>, <||>
|
||||||
|
|
||||||
|
-- | 'p <&&> q'. '&&' lifted to a Monad.
|
||||||
|
(<&&>) :: Monad m => m Bool -> m Bool -> m Bool
|
||||||
|
(<&&>) = liftM2 (&&)
|
||||||
|
|
||||||
|
-- | 'p <||> q'. '||' lifted to a Monad.
|
||||||
|
(<||>) :: Monad m => m Bool -> m Bool -> m Bool
|
||||||
|
(<||>) = liftM2 (||)
|
||||||
|
|
||||||
|
-- | Queries that return the window title, resource, or class.
|
||||||
|
title, resource, className :: Query String
|
||||||
|
title = ask >>= (\w -> liftX $ withDisplay $ \d -> fmap (fromMaybe "") $ io $ fetchName d w)
|
||||||
|
resource = ask >>= (\w -> liftX $ withDisplay $ \d -> fmap resName $ io $ getClassHint d w)
|
||||||
|
className = ask >>= (\w -> liftX $ withDisplay $ \d -> fmap resClass $ io $ getClassHint d w)
|
||||||
|
|
||||||
|
-- | Modify the 'WindowSet' with a pure function.
|
||||||
|
doF :: (WindowSet -> WindowSet) -> ManageHook
|
||||||
|
doF = return . Endo
|
||||||
|
|
||||||
|
-- | Move the window to the floating layer.
|
||||||
|
doFloat :: ManageHook
|
||||||
|
doFloat = ask >>= \w -> doF . W.float w . snd =<< liftX (floatLocation w)
|
||||||
|
|
||||||
|
-- | Map the window and remove it from the 'WindowSet'.
|
||||||
|
doIgnore :: ManageHook
|
||||||
|
doIgnore = ask >>= \w -> liftX (reveal w) >> doF (W.delete w)
|
@@ -1,11 +1,13 @@
|
|||||||
{-# OPTIONS -fno-warn-orphans -fglasgow-exts #-}
|
{-# OPTIONS_GHC -fno-warn-orphans #-}
|
||||||
-- \^^ deriving Typeable
|
{-# OPTIONS_GHC -fglasgow-exts #-} -- For deriving Data/Typeable
|
||||||
|
{-# LANGUAGE FlexibleInstances, MultiParamTypeClasses, PatternGuards, TypeSynonymInstances #-}
|
||||||
|
|
||||||
-- --------------------------------------------------------------------------
|
-- --------------------------------------------------------------------------
|
||||||
-- |
|
-- |
|
||||||
-- Module : Operations.hs
|
-- Module : XMonad.Operations
|
||||||
-- Copyright : (c) Spencer Janssen 2007
|
-- Copyright : (c) Spencer Janssen 2007
|
||||||
-- License : BSD3-style (see LICENSE)
|
-- License : BSD3-style (see LICENSE)
|
||||||
--
|
--
|
||||||
-- Maintainer : dons@cse.unsw.edu.au
|
-- Maintainer : dons@cse.unsw.edu.au
|
||||||
-- Stability : unstable
|
-- Stability : unstable
|
||||||
-- Portability : not portable, Typeable deriving, mtl, posix
|
-- Portability : not portable, Typeable deriving, mtl, posix
|
||||||
@@ -14,11 +16,11 @@
|
|||||||
--
|
--
|
||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
module Operations where
|
module XMonad.Operations where
|
||||||
|
|
||||||
import XMonad
|
import XMonad.Core
|
||||||
import qualified StackSet as W
|
import XMonad.Layout (Full(..))
|
||||||
import {-# SOURCE #-} Config (borderWidth,logHook,numlockMask)
|
import qualified XMonad.StackSet as W
|
||||||
|
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.List (nub, (\\), find)
|
import Data.List (nub, (\\), find)
|
||||||
@@ -27,17 +29,15 @@ import Data.Ratio
|
|||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
import qualified Data.Set as S
|
import qualified Data.Set as S
|
||||||
|
|
||||||
|
import Control.Applicative
|
||||||
import Control.Monad.State
|
import Control.Monad.State
|
||||||
import Control.Monad.Reader
|
import Control.Monad.Reader
|
||||||
import Control.Arrow ((***), first, second)
|
|
||||||
|
|
||||||
import System.IO
|
import System.IO
|
||||||
import Graphics.X11.Xlib
|
import Graphics.X11.Xlib
|
||||||
import Graphics.X11.Xinerama (getScreenInfo)
|
import Graphics.X11.Xinerama (getScreenInfo)
|
||||||
import Graphics.X11.Xlib.Extras
|
import Graphics.X11.Xlib.Extras
|
||||||
|
|
||||||
import qualified Data.Traversable as T
|
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- |
|
-- |
|
||||||
-- Window manager operations
|
-- Window manager operations
|
||||||
@@ -48,51 +48,31 @@ import qualified Data.Traversable as T
|
|||||||
-- border set, and its event mask set.
|
-- border set, and its event mask set.
|
||||||
--
|
--
|
||||||
manage :: Window -> X ()
|
manage :: Window -> X ()
|
||||||
manage w = whenX (fmap not $ isClient w) $ withDisplay $ \d -> do
|
manage w = whenX (not <$> isClient w) $ withDisplay $ \d -> do
|
||||||
setInitialProperties w >> reveal w
|
|
||||||
|
|
||||||
-- FIXME: This is pretty awkward. We can't can't let "refresh" happen
|
|
||||||
-- before the call to float, because that will resize the window and
|
|
||||||
-- lose the default sizing.
|
|
||||||
|
|
||||||
sh <- io $ getWMNormalHints d w
|
sh <- io $ getWMNormalHints d w
|
||||||
|
|
||||||
let isFixedSize = sh_min_size sh /= Nothing && sh_min_size sh == sh_max_size sh
|
let isFixedSize = sh_min_size sh /= Nothing && sh_min_size sh == sh_max_size sh
|
||||||
isTransient <- isJust `liftM` io (getTransientForHint d w)
|
isTransient <- isJust <$> io (getTransientForHint d w)
|
||||||
if isFixedSize || isTransient
|
|
||||||
then do modify $ \s -> s { windowset = W.insertUp w (windowset s) }
|
(sc, rr) <- floatLocation w
|
||||||
float w -- \^^ now go the refresh.
|
-- ensure that float windows don't go over the edge of the screen
|
||||||
else windows $ W.insertUp w
|
let adjust (W.RationalRect x y wid h) | x + wid > 1 || y + h > 1 || x < 0 || y < 0
|
||||||
|
= W.RationalRect (0.5 - wid/2) (0.5 - h/2) wid h
|
||||||
|
adjust r = r
|
||||||
|
|
||||||
|
f ws | isFixedSize || isTransient = W.float w (adjust rr) . W.insertUp w . W.view i $ ws
|
||||||
|
| otherwise = W.insertUp w ws
|
||||||
|
where i = fromMaybe (W.tag . W.workspace . W.current $ ws) $ W.lookupWorkspace sc ws
|
||||||
|
|
||||||
|
mh <- asks (manageHook . config)
|
||||||
|
g <- runManageHook mh w `catchX` return id
|
||||||
|
windows (g . f)
|
||||||
|
|
||||||
-- | unmanage. A window no longer exists, remove it from the window
|
-- | unmanage. A window no longer exists, remove it from the window
|
||||||
-- list, on whatever workspace it is.
|
-- list, on whatever workspace it is.
|
||||||
--
|
--
|
||||||
-- FIXME: clearFloating should be taken care of in W.delete, but if we do it
|
|
||||||
-- there, floating status is lost when moving windows between workspaces,
|
|
||||||
-- because W.shift calls W.delete.
|
|
||||||
--
|
|
||||||
-- should also unmap?
|
|
||||||
--
|
|
||||||
unmanage :: Window -> X ()
|
unmanage :: Window -> X ()
|
||||||
unmanage w = do
|
unmanage = windows . W.delete
|
||||||
windows (W.sink w . W.delete w)
|
|
||||||
setWMState w 0 {-withdrawn-}
|
|
||||||
modify (\s -> s {mapped = S.delete w (mapped s), waitingUnmap = M.delete w (waitingUnmap s)})
|
|
||||||
|
|
||||||
-- | focus. focus window up or down. or swap various windows.
|
|
||||||
focusUp, focusDown, swapUp, swapDown, swapMaster :: X ()
|
|
||||||
focusUp = windows W.focusUp
|
|
||||||
focusDown = windows W.focusDown
|
|
||||||
swapUp = windows W.swapUp
|
|
||||||
swapDown = windows W.swapDown
|
|
||||||
swapMaster = windows W.swapMaster
|
|
||||||
|
|
||||||
-- | shift. Move a window to a new workspace, 0 indexed.
|
|
||||||
shift :: WorkspaceId -> X ()
|
|
||||||
shift n = windows (W.shift n)
|
|
||||||
|
|
||||||
-- | view. Change the current workspace to workspace at offset n (0 indexed).
|
|
||||||
view :: WorkspaceId -> X ()
|
|
||||||
view = windows . W.greedyView
|
|
||||||
|
|
||||||
-- | Modify the size of the status gap at the top of the current screen
|
-- | Modify the size of the status gap at the top of the current screen
|
||||||
-- Taking a function giving the current screen, and current geometry.
|
-- Taking a function giving the current screen, and current geometry.
|
||||||
@@ -124,33 +104,35 @@ kill = withDisplay $ \d -> withFocused $ \w -> do
|
|||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- Managing windows
|
-- Managing windows
|
||||||
|
|
||||||
data UnDoLayout = UnDoLayout deriving ( Typeable, Eq )
|
|
||||||
instance Message UnDoLayout
|
|
||||||
|
|
||||||
-- | windows. Modify the current window list with a pure function, and refresh
|
-- | windows. Modify the current window list with a pure function, and refresh
|
||||||
windows :: (WindowSet -> WindowSet) -> X ()
|
windows :: (WindowSet -> WindowSet) -> X ()
|
||||||
windows f = do
|
windows f = do
|
||||||
-- Notify visible layouts to remove decorations etc
|
XState { windowset = old } <- get
|
||||||
-- We cannot use sendMessage because this must not call refresh ever,
|
|
||||||
-- and must be called on all visible workspaces.
|
|
||||||
broadcastMessage UnDoLayout
|
|
||||||
XState { windowset = old, layouts = fls } <- get
|
|
||||||
let oldvisible = concatMap (W.integrate' . W.stack . W.workspace) $ W.current old : W.visible old
|
let oldvisible = concatMap (W.integrate' . W.stack . W.workspace) $ W.current old : W.visible old
|
||||||
ws = f old
|
ws = f old
|
||||||
|
XConf { display = d , normalBorder = nbc, focusedBorder = fbc } <- ask
|
||||||
|
|
||||||
|
mapM_ setInitialProperties (W.allWindows ws \\ W.allWindows old)
|
||||||
|
|
||||||
|
whenJust (W.peek old) $ \otherw -> io $ setWindowBorder d otherw nbc
|
||||||
modify (\s -> s { windowset = ws })
|
modify (\s -> s { windowset = ws })
|
||||||
d <- asks display
|
|
||||||
|
-- notify non visibility
|
||||||
|
let tags_oldvisible = map (W.tag . W.workspace) $ W.current old : W.visible old
|
||||||
|
gottenhidden = filter (`elem` tags_oldvisible) $ map W.tag $ W.hidden ws
|
||||||
|
sendMessageToWorkspaces Hide gottenhidden
|
||||||
|
|
||||||
-- for each workspace, layout the currently visible workspaces
|
-- for each workspace, layout the currently visible workspaces
|
||||||
let allscreens = W.current ws : W.visible ws
|
let allscreens = W.screens ws
|
||||||
summed_visible = scanl (++) [] $ map (W.integrate' . W.stack . W.workspace) allscreens
|
summed_visible = scanl (++) [] $ map (W.integrate' . W.stack . W.workspace) allscreens
|
||||||
visible <- fmap concat $ forM (zip allscreens summed_visible) $ \ (w, vis) -> do
|
visible <- fmap concat $ forM (zip allscreens summed_visible) $ \ (w, vis) -> do
|
||||||
let n = W.tag (W.workspace w)
|
let n = W.tag (W.workspace w)
|
||||||
this = W.view n ws
|
this = W.view n ws
|
||||||
Just l = fmap fst $ M.lookup n fls
|
l = W.layout (W.workspace w)
|
||||||
flt = filter (flip M.member (W.floating ws)) (W.index this)
|
flt = filter (flip M.member (W.floating ws)) (W.index this)
|
||||||
tiled = (W.stack . W.workspace . W.current $ this)
|
tiled = (W.stack . W.workspace . W.current $ this)
|
||||||
>>= W.filter (not . flip M.member (W.floating ws))
|
>>= W.filter (`M.notMember` W.floating ws)
|
||||||
>>= W.filter (not . (`elem` vis))
|
>>= W.filter (`notElem` vis)
|
||||||
(SD (Rectangle sx sy sw sh)
|
(SD (Rectangle sx sy sw sh)
|
||||||
(gt,gb,gl,gr)) = W.screenDetail w
|
(gt,gb,gl,gr)) = W.screenDetail w
|
||||||
viewrect = Rectangle (sx + fromIntegral gl) (sy + fromIntegral gt)
|
viewrect = Rectangle (sx + fromIntegral gl) (sy + fromIntegral gt)
|
||||||
@@ -158,10 +140,11 @@ windows f = do
|
|||||||
|
|
||||||
-- just the tiled windows:
|
-- just the tiled windows:
|
||||||
-- now tile the windows on this workspace, modified by the gap
|
-- now tile the windows on this workspace, modified by the gap
|
||||||
(rs, ml') <- runLayout l viewrect tiled `catchX` runLayout full viewrect tiled
|
(rs, ml') <- runLayout l viewrect tiled `catchX` runLayout (Layout Full) viewrect tiled
|
||||||
mapM_ (uncurry tileWindow) rs
|
mapM_ (uncurry tileWindow) rs
|
||||||
whenJust ml' $ \l' -> modify $ \ss ->
|
whenJust ml' $ \l' -> runOnWorkspaces (\ww -> if W.tag ww == n
|
||||||
ss { layouts = M.adjust (first (const l')) n (layouts ss) }
|
then return $ ww { W.layout = l'}
|
||||||
|
else return ww)
|
||||||
|
|
||||||
-- now the floating windows:
|
-- now the floating windows:
|
||||||
-- move/resize the floating windows, if there are any
|
-- move/resize the floating windows, if there are any
|
||||||
@@ -176,14 +159,19 @@ windows f = do
|
|||||||
-- return the visible windows for this workspace:
|
-- return the visible windows for this workspace:
|
||||||
return vs
|
return vs
|
||||||
|
|
||||||
|
whenJust (W.peek ws) $ \w -> io $ setWindowBorder d w fbc
|
||||||
setTopFocus
|
setTopFocus
|
||||||
logHook
|
asks (logHook . config) >>= userCode
|
||||||
-- io performGC -- really helps, but seems to trigger GC bugs?
|
|
||||||
|
|
||||||
-- hide every window that was potentially visible before, but is not
|
-- hide every window that was potentially visible before, but is not
|
||||||
-- given a position by a layout now.
|
-- given a position by a layout now.
|
||||||
mapM_ hide (nub oldvisible \\ visible)
|
mapM_ hide (nub oldvisible \\ visible)
|
||||||
|
|
||||||
|
-- all windows that are no longer in the windowset are marked as
|
||||||
|
-- withdrawn, it is important to do this after the above, otherwise 'hide'
|
||||||
|
-- will overwrite withdrawnState with iconicState
|
||||||
|
mapM_ (flip setWMState withdrawnState) (W.allWindows old \\ W.allWindows ws)
|
||||||
|
|
||||||
clearEvents enterWindowMask
|
clearEvents enterWindowMask
|
||||||
|
|
||||||
-- | setWMState. set the WM_STATE property
|
-- | setWMState. set the WM_STATE property
|
||||||
@@ -198,7 +186,7 @@ hide w = whenX (gets (S.member w . mapped)) $ withDisplay $ \d -> do
|
|||||||
io $ do selectInput d w (clientMask .&. complement structureNotifyMask)
|
io $ do selectInput d w (clientMask .&. complement structureNotifyMask)
|
||||||
unmapWindow d w
|
unmapWindow d w
|
||||||
selectInput d w clientMask
|
selectInput d w clientMask
|
||||||
setWMState w 3 --iconic
|
setWMState w iconicState
|
||||||
-- this part is key: we increment the waitingUnmap counter to distinguish
|
-- this part is key: we increment the waitingUnmap counter to distinguish
|
||||||
-- between client and xmonad initiated unmaps.
|
-- between client and xmonad initiated unmaps.
|
||||||
modify (\s -> s { waitingUnmap = M.insertWith (+) w 1 (waitingUnmap s)
|
modify (\s -> s { waitingUnmap = M.insertWith (+) w 1 (waitingUnmap s)
|
||||||
@@ -208,7 +196,7 @@ hide w = whenX (gets (S.member w . mapped)) $ withDisplay $ \d -> do
|
|||||||
-- this is harmless if the window was already visible
|
-- this is harmless if the window was already visible
|
||||||
reveal :: Window -> X ()
|
reveal :: Window -> X ()
|
||||||
reveal w = withDisplay $ \d -> do
|
reveal w = withDisplay $ \d -> do
|
||||||
setWMState w 1 --normal
|
setWMState w normalState
|
||||||
io $ mapWindow d w
|
io $ mapWindow d w
|
||||||
modify (\s -> s { mapped = S.insert w (mapped s) })
|
modify (\s -> s { mapped = S.insert w (mapped s) })
|
||||||
|
|
||||||
@@ -218,9 +206,14 @@ clientMask = structureNotifyMask .|. enterWindowMask .|. propertyChangeMask
|
|||||||
|
|
||||||
-- | Set some properties when we initially gain control of a window
|
-- | Set some properties when we initially gain control of a window
|
||||||
setInitialProperties :: Window -> X ()
|
setInitialProperties :: Window -> X ()
|
||||||
setInitialProperties w = withDisplay $ \d -> io $ do
|
setInitialProperties w = asks normalBorder >>= \nb -> withDisplay $ \d -> do
|
||||||
selectInput d w $ clientMask
|
setWMState w iconicState
|
||||||
setWindowBorderWidth d w borderWidth
|
io $ selectInput d w $ clientMask
|
||||||
|
bw <- asks (borderWidth . config)
|
||||||
|
io $ setWindowBorderWidth d w bw
|
||||||
|
-- we must initially set the color of new windows, to maintain invariants
|
||||||
|
-- required by the border setting in 'windows'
|
||||||
|
io $ setWindowBorder d w nb
|
||||||
|
|
||||||
-- | refresh. Render the currently visible workspaces, as determined by
|
-- | refresh. Render the currently visible workspaces, as determined by
|
||||||
-- the StackSet. Also, set focus to the focused window.
|
-- the StackSet. Also, set focus to the focused window.
|
||||||
@@ -243,7 +236,7 @@ clearEvents mask = withDisplay $ \d -> io $ do
|
|||||||
-- rectangle, including its border.
|
-- rectangle, including its border.
|
||||||
tileWindow :: Window -> Rectangle -> X ()
|
tileWindow :: Window -> Rectangle -> X ()
|
||||||
tileWindow w r = withDisplay $ \d -> do
|
tileWindow w r = withDisplay $ \d -> do
|
||||||
bw <- (fromIntegral . wa_border_width) `liftM` io (getWindowAttributes d w)
|
bw <- (fromIntegral . wa_border_width) <$> io (getWindowAttributes d w)
|
||||||
-- give all windows at least 1x1 pixels
|
-- give all windows at least 1x1 pixels
|
||||||
let least x | x <= bw*2 = 1
|
let least x | x <= bw*2 = 1
|
||||||
| otherwise = x - bw*2
|
| otherwise = x - bw*2
|
||||||
@@ -253,11 +246,31 @@ tileWindow w r = withDisplay $ \d -> do
|
|||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- | Returns True if the first rectangle is contained within, but not equal
|
||||||
|
-- to the second.
|
||||||
|
containedIn :: Rectangle -> Rectangle -> Bool
|
||||||
|
containedIn r1@(Rectangle x1 y1 w1 h1) r2@(Rectangle x2 y2 w2 h2)
|
||||||
|
= and [ r1 /= r2
|
||||||
|
, x1 >= x2
|
||||||
|
, y1 >= y2
|
||||||
|
, fromIntegral x1 + w1 <= fromIntegral x2 + w2
|
||||||
|
, fromIntegral y1 + h1 <= fromIntegral y2 + h2 ]
|
||||||
|
|
||||||
|
-- | Given a list of screens, remove all duplicated screens and screens that
|
||||||
|
-- are entirely contained within another.
|
||||||
|
nubScreens :: [Rectangle] -> [Rectangle]
|
||||||
|
nubScreens xs = nub . filter (\x -> not $ any (x `containedIn`) xs) $ xs
|
||||||
|
|
||||||
|
-- | Cleans the list of screens according to the rules documented for
|
||||||
|
-- nubScreens.
|
||||||
|
getCleanedScreenInfo :: MonadIO m => Display -> m [Rectangle]
|
||||||
|
getCleanedScreenInfo = io . fmap nubScreens . getScreenInfo
|
||||||
|
|
||||||
-- | rescreen. The screen configuration may have changed (due to
|
-- | rescreen. The screen configuration may have changed (due to
|
||||||
-- xrandr), update the state and refresh the screen, and reset the gap.
|
-- xrandr), update the state and refresh the screen, and reset the gap.
|
||||||
rescreen :: X ()
|
rescreen :: X ()
|
||||||
rescreen = do
|
rescreen = do
|
||||||
xinesc <- withDisplay (io . getScreenInfo)
|
xinesc <- withDisplay getCleanedScreenInfo
|
||||||
|
|
||||||
windows $ \ws@(W.StackSet { W.current = v, W.visible = vs, W.hidden = hs }) ->
|
windows $ \ws@(W.StackSet { W.current = v, W.visible = vs, W.hidden = hs }) ->
|
||||||
let (xs, ys) = splitAt (length xinesc) $ map W.workspace (v:vs) ++ hs
|
let (xs, ys) = splitAt (length xinesc) $ map W.workspace (v:vs) ++ hs
|
||||||
@@ -297,147 +310,46 @@ focus w = withWindowSet $ \s -> do
|
|||||||
-- | Call X to set the keyboard focus details.
|
-- | Call X to set the keyboard focus details.
|
||||||
setFocusX :: Window -> X ()
|
setFocusX :: Window -> X ()
|
||||||
setFocusX w = withWindowSet $ \ws -> do
|
setFocusX w = withWindowSet $ \ws -> do
|
||||||
XConf { display = dpy , normalBorder = nbc, focusedBorder = fbc } <- ask
|
dpy <- asks display
|
||||||
|
|
||||||
-- clear mouse button grab and border on other windows
|
-- clear mouse button grab and border on other windows
|
||||||
forM_ (W.current ws : W.visible ws) $ \wk -> do
|
forM_ (W.current ws : W.visible ws) $ \wk -> do
|
||||||
forM_ (W.index (W.view (W.tag (W.workspace wk)) ws)) $ \otherw -> do
|
forM_ (W.index (W.view (W.tag (W.workspace wk)) ws)) $ \otherw -> do
|
||||||
setButtonGrab True otherw
|
setButtonGrab True otherw
|
||||||
io $ setWindowBorder dpy otherw nbc
|
|
||||||
|
|
||||||
-- If we ungrab buttons on the root window, we lose our mouse bindings.
|
-- If we ungrab buttons on the root window, we lose our mouse bindings.
|
||||||
whenX (not `liftM` isRoot w) $ setButtonGrab False w
|
whenX (not <$> isRoot w) $ setButtonGrab False w
|
||||||
io $ do setInputFocus dpy w revertToPointerRoot 0
|
io $ do setInputFocus dpy w revertToPointerRoot 0
|
||||||
-- raiseWindow dpy w
|
-- raiseWindow dpy w
|
||||||
io $ setWindowBorder dpy w fbc
|
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- Managing layout
|
-- Message handling
|
||||||
|
|
||||||
-- | switchLayout. Switch to another layout scheme. Switches the
|
-- | Throw a message to the current LayoutClass possibly modifying how we
|
||||||
-- layout of the current workspace. By convention, a window set as
|
|
||||||
-- master in Tall mode remains as master in Wide mode. When switching
|
|
||||||
-- from full screen to a tiling mode, the currently focused window
|
|
||||||
-- becomes a master. When switching back , the focused window is
|
|
||||||
-- uppermost.
|
|
||||||
--
|
|
||||||
-- Note that the new layout's deconstructor will be called, so it should be
|
|
||||||
-- idempotent.
|
|
||||||
switchLayout :: X ()
|
|
||||||
switchLayout = do
|
|
||||||
broadcastMessage UnDoLayout -- calling refresh now would defeat the point of deconstruction
|
|
||||||
n <- gets (W.tag . W.workspace . W.current . windowset)
|
|
||||||
modify $ \s -> s { layouts = M.adjust switch n (layouts s) }
|
|
||||||
refresh
|
|
||||||
where switch (x, xs) = let xs' = xs ++ [x] in (head xs', tail xs')
|
|
||||||
|
|
||||||
-- | Throw a message to the current Layout possibly modifying how we
|
|
||||||
-- layout the windows, then refresh.
|
-- layout the windows, then refresh.
|
||||||
--
|
|
||||||
sendMessage :: Message a => a -> X ()
|
sendMessage :: Message a => a -> X ()
|
||||||
sendMessage a = do n <- (W.tag . W.workspace . W.current) `fmap` gets windowset
|
sendMessage a = do
|
||||||
Just (l,ls) <- M.lookup n `fmap` gets layouts
|
w <- W.workspace . W.current <$> gets windowset
|
||||||
ml' <- modifyLayout l (SomeMessage a) `catchX` return (Just l)
|
ml' <- handleMessage (W.layout w) (SomeMessage a) `catchX` return Nothing
|
||||||
whenJust ml' $ \l' -> do modify $ \s -> s { layouts = M.insert n (l',ls) (layouts s) }
|
whenJust ml' $ \l' -> do
|
||||||
refresh
|
windows $ \ws -> ws { W.current = (W.current ws)
|
||||||
|
{ W.workspace = (W.workspace $ W.current ws)
|
||||||
|
{ W.layout = l' }}}
|
||||||
|
|
||||||
-- | Send a message to all visible layouts, without necessarily refreshing.
|
-- | Send a message to a list of workspaces' layouts, without necessarily refreshing.
|
||||||
-- This is how we implement the hooks, such as UnDoLayout.
|
sendMessageToWorkspaces :: Message a => a -> [WorkspaceId] -> X ()
|
||||||
broadcastMessage :: Message a => a -> X ()
|
sendMessageToWorkspaces a l = runOnWorkspaces $ \w ->
|
||||||
broadcastMessage a = do
|
if W.tag w `elem` l
|
||||||
ol <- gets layouts
|
then do ml' <- handleMessage (W.layout w) (SomeMessage a) `catchX` return Nothing
|
||||||
nl <- T.forM ol $ \ (l,ls) -> maybe (l,ls) (flip (,) ls) `fmap`
|
return $ w { W.layout = maybe (W.layout w) id ml' }
|
||||||
(modifyLayout l (SomeMessage a) `catchX` return (Just l))
|
else return w
|
||||||
modify $ \s -> s { layouts = nl }
|
|
||||||
|
|
||||||
instance Message Event
|
-- | Set the layout of the currently viewed workspace
|
||||||
|
setLayout :: Layout Window -> X ()
|
||||||
--
|
setLayout l = do
|
||||||
-- Builtin layout algorithms:
|
ss@(W.StackSet { W.current = c@(W.Screen { W.workspace = ws })}) <- gets windowset
|
||||||
--
|
handleMessage (W.layout ws) (SomeMessage ReleaseResources)
|
||||||
-- fullscreen mode
|
windows $ const $ ss {W.current = c { W.workspace = ws { W.layout = l } } }
|
||||||
-- tall mode
|
|
||||||
--
|
|
||||||
-- The latter algorithms support the following operations:
|
|
||||||
--
|
|
||||||
-- Shrink
|
|
||||||
-- Expand
|
|
||||||
--
|
|
||||||
|
|
||||||
data Resize = Shrink | Expand deriving Typeable
|
|
||||||
data IncMasterN = IncMasterN Int deriving Typeable
|
|
||||||
instance Message Resize
|
|
||||||
instance Message IncMasterN
|
|
||||||
|
|
||||||
-- simple fullscreen mode, just render all windows fullscreen.
|
|
||||||
-- a plea for tuple sections: map . (,sc)
|
|
||||||
full :: Layout a
|
|
||||||
full = Layout { doLayout = \sc (W.Stack f _ _) -> return ([(f, sc)],Nothing)
|
|
||||||
, modifyLayout = const (return Nothing) } -- no changes
|
|
||||||
|
|
||||||
--
|
|
||||||
-- The tiling mode of xmonad, and its operations.
|
|
||||||
--
|
|
||||||
tall :: Int -> Rational -> Rational -> Layout a
|
|
||||||
tall nmaster delta frac =
|
|
||||||
Layout { doLayout = \r -> return . (\x->(x,Nothing)) .
|
|
||||||
ap zip (tile frac r nmaster . length) . W.integrate
|
|
||||||
, modifyLayout = \m -> return $ msum [fmap resize (fromMessage m)
|
|
||||||
,fmap incmastern (fromMessage m)] }
|
|
||||||
|
|
||||||
where resize Shrink = tall nmaster delta (max 0 $ frac-delta)
|
|
||||||
resize Expand = tall nmaster delta (min 1 $ frac+delta)
|
|
||||||
incmastern (IncMasterN d) = tall (max 0 (nmaster+d)) delta frac
|
|
||||||
|
|
||||||
-- | Mirror a rectangle
|
|
||||||
mirrorRect :: Rectangle -> Rectangle
|
|
||||||
mirrorRect (Rectangle rx ry rw rh) = (Rectangle ry rx rh rw)
|
|
||||||
|
|
||||||
-- | Mirror a layout, compute its 90 degree rotated form.
|
|
||||||
mirror :: Layout a -> Layout a
|
|
||||||
mirror (Layout { doLayout = dl, modifyLayout = ml }) =
|
|
||||||
Layout { doLayout = \sc w -> do (wrs, ml') <- dl (mirrorRect sc) w
|
|
||||||
return (map (second mirrorRect) wrs, mirror `fmap` ml')
|
|
||||||
, modifyLayout = fmap (fmap mirror) . ml }
|
|
||||||
|
|
||||||
-- | tile. Compute the positions for windows using the default 2 pane tiling algorithm.
|
|
||||||
--
|
|
||||||
-- The screen is divided (currently) into two panes. all clients are
|
|
||||||
-- then partioned between these two panes. one pane, the `master', by
|
|
||||||
-- convention has the least number of windows in it (by default, 1).
|
|
||||||
-- the variable `nmaster' controls how many windows are rendered in the
|
|
||||||
-- master pane.
|
|
||||||
--
|
|
||||||
-- `delta' specifies the ratio of the screen to resize by.
|
|
||||||
--
|
|
||||||
-- 'frac' specifies what proportion of the screen to devote to the
|
|
||||||
-- master area.
|
|
||||||
--
|
|
||||||
tile :: Rational -> Rectangle -> Int -> Int -> [Rectangle]
|
|
||||||
tile f r nmaster n = if n <= nmaster || nmaster == 0
|
|
||||||
then splitVertically n r
|
|
||||||
else splitVertically nmaster r1 ++ splitVertically (n-nmaster) r2 -- two columns
|
|
||||||
where (r1,r2) = splitHorizontallyBy f r
|
|
||||||
|
|
||||||
--
|
|
||||||
-- Divide the screen vertically into n subrectangles
|
|
||||||
--
|
|
||||||
splitVertically, splitHorizontally :: Int -> Rectangle -> [Rectangle]
|
|
||||||
splitVertically n r | n < 2 = [r]
|
|
||||||
splitVertically n (Rectangle sx sy sw sh) = Rectangle sx sy sw smallh :
|
|
||||||
splitVertically (n-1) (Rectangle sx (sy+fromIntegral smallh) sw (sh-smallh))
|
|
||||||
where smallh = sh `div` fromIntegral n --hmm, this is a fold or map.
|
|
||||||
|
|
||||||
splitHorizontally n = map mirrorRect . splitVertically n . mirrorRect
|
|
||||||
|
|
||||||
-- Divide the screen into two rectangles, using a rational to specify the ratio
|
|
||||||
splitHorizontallyBy, splitVerticallyBy :: RealFrac r => r -> Rectangle -> (Rectangle, Rectangle)
|
|
||||||
splitHorizontallyBy f (Rectangle sx sy sw sh) =
|
|
||||||
( Rectangle sx sy leftw sh
|
|
||||||
, Rectangle (sx + fromIntegral leftw) sy (sw-fromIntegral leftw) sh)
|
|
||||||
where leftw = floor $ fromIntegral sw * f
|
|
||||||
|
|
||||||
splitVerticallyBy f = (mirrorRect *** mirrorRect) . splitHorizontallyBy f . mirrorRect
|
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- Utilities
|
-- Utilities
|
||||||
@@ -456,45 +368,42 @@ isClient w = withWindowSet $ return . W.member w
|
|||||||
|
|
||||||
-- | Combinations of extra modifier masks we need to grab keys\/buttons for.
|
-- | Combinations of extra modifier masks we need to grab keys\/buttons for.
|
||||||
-- (numlock and capslock)
|
-- (numlock and capslock)
|
||||||
extraModifiers :: [KeyMask]
|
extraModifiers :: X [KeyMask]
|
||||||
extraModifiers = [0, numlockMask, lockMask, numlockMask .|. lockMask ]
|
extraModifiers = do
|
||||||
|
nlm <- asks (numlockMask . config)
|
||||||
|
return [0, nlm, lockMask, nlm .|. lockMask ]
|
||||||
|
|
||||||
-- | Strip numlock\/capslock from a mask
|
-- | Strip numlock\/capslock from a mask
|
||||||
cleanMask :: KeyMask -> KeyMask
|
cleanMask :: KeyMask -> X KeyMask
|
||||||
cleanMask = (complement (numlockMask .|. lockMask) .&.)
|
cleanMask km = do
|
||||||
|
nlm <- asks (numlockMask . config)
|
||||||
|
return (complement (nlm .|. lockMask) .&. km)
|
||||||
|
|
||||||
-- | Get the Pixel value for a named color
|
-- | Get the Pixel value for a named color
|
||||||
initColor :: Display -> String -> IO Pixel
|
initColor :: Display -> String -> IO Pixel
|
||||||
initColor dpy c = (color_pixel . fst) `liftM` allocNamedColor dpy colormap c
|
initColor dpy c = (color_pixel . fst) <$> allocNamedColor dpy colormap c
|
||||||
where colormap = defaultColormap dpy (defaultScreen dpy)
|
where colormap = defaultColormap dpy (defaultScreen dpy)
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- | Floating layer support
|
-- | Floating layer support
|
||||||
|
|
||||||
-- | Make a floating window tiled
|
-- | Given a window, find the screen it is located on, and compute
|
||||||
sink :: Window -> X ()
|
-- the geometry of that window wrt. that screen.
|
||||||
sink = windows . W.sink
|
floatLocation :: Window -> X (ScreenId, W.RationalRect)
|
||||||
|
floatLocation w = withDisplay $ \d -> do
|
||||||
-- | Make a tiled window floating, using its suggested rectangle
|
|
||||||
--
|
|
||||||
-- TODO: float changes the set of visible workspaces when we call it for an
|
|
||||||
-- invisible window -- this should not happen. See 'temporary workaround' in
|
|
||||||
-- the handler for ConfigureRequestEvent also.
|
|
||||||
float :: Window -> X ()
|
|
||||||
float w = withDisplay $ \d -> do
|
|
||||||
ws <- gets windowset
|
ws <- gets windowset
|
||||||
wa <- io $ getWindowAttributes d w
|
wa <- io $ getWindowAttributes d w
|
||||||
|
bw <- fi <$> asks (borderWidth . config)
|
||||||
|
|
||||||
let sc = fromMaybe (W.current ws) $ find (pointWithin (fi $ wa_x wa) (fi $ wa_y wa) . screenRect . W.screenDetail) $ W.current ws : W.visible ws
|
-- XXX horrible
|
||||||
|
let sc = fromMaybe (W.current ws) $ find (pointWithin (fi $ wa_x wa) (fi $ wa_y wa) . screenRect . W.screenDetail) $ W.screens ws
|
||||||
sr = screenRect . W.screenDetail $ sc
|
sr = screenRect . W.screenDetail $ sc
|
||||||
sw = W.tag . W.workspace $ sc
|
rr = W.RationalRect ((fi (wa_x wa) - fi (rect_x sr)) % fi (rect_width sr))
|
||||||
bw = fi . wa_border_width $ wa
|
((fi (wa_y wa) - fi (rect_y sr)) % fi (rect_height sr))
|
||||||
|
(fi (wa_width wa + bw*2) % fi (rect_width sr))
|
||||||
|
(fi (wa_height wa + bw*2) % fi (rect_height sr))
|
||||||
|
|
||||||
windows $ maybe id W.focusWindow (W.peek ws) . W.shift sw . W.focusWindow w . W.float w
|
return (W.screen $ sc, rr)
|
||||||
(W.RationalRect ((fi (wa_x wa) - fi (rect_x sr)) % fi (rect_width sr))
|
|
||||||
((fi (wa_y wa) - fi (rect_y sr)) % fi (rect_height sr))
|
|
||||||
(fi (wa_width wa + bw*2) % fi (rect_width sr))
|
|
||||||
(fi (wa_height wa + bw*2) % fi (rect_height sr)))
|
|
||||||
where fi x = fromIntegral x
|
where fi x = fromIntegral x
|
||||||
pointWithin :: Integer -> Integer -> Rectangle -> Bool
|
pointWithin :: Integer -> Integer -> Rectangle -> Bool
|
||||||
pointWithin x y r = x >= fi (rect_x r) &&
|
pointWithin x y r = x >= fi (rect_x r) &&
|
||||||
@@ -502,6 +411,17 @@ float w = withDisplay $ \d -> do
|
|||||||
y >= fi (rect_y r) &&
|
y >= fi (rect_y r) &&
|
||||||
y < fi (rect_y r) + fi (rect_height r)
|
y < fi (rect_y r) + fi (rect_height r)
|
||||||
|
|
||||||
|
-- | Make a tiled window floating, using its suggested rectangle
|
||||||
|
float :: Window -> X ()
|
||||||
|
float w = do
|
||||||
|
(sc, rr) <- floatLocation w
|
||||||
|
windows $ \ws -> W.float w rr . fromMaybe ws $ do
|
||||||
|
i <- W.findTag w ws
|
||||||
|
guard $ i `elem` map (W.tag . W.workspace) (W.screens ws)
|
||||||
|
f <- W.peek ws
|
||||||
|
sw <- W.lookupWorkspace sc ws
|
||||||
|
return (W.focusWindow f . W.shiftWin sw w $ ws)
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- Mouse handling
|
-- Mouse handling
|
||||||
|
|
||||||
@@ -525,6 +445,7 @@ mouseDrag f done = do
|
|||||||
clearEvents pointerMotionMask
|
clearEvents pointerMotionMask
|
||||||
return z
|
return z
|
||||||
|
|
||||||
|
-- | XXX comment me
|
||||||
mouseMoveWindow :: Window -> X ()
|
mouseMoveWindow :: Window -> X ()
|
||||||
mouseMoveWindow w = whenX (isClient w) $ withDisplay $ \d -> do
|
mouseMoveWindow w = whenX (isClient w) $ withDisplay $ \d -> do
|
||||||
io $ raiseWindow d w
|
io $ raiseWindow d w
|
||||||
@@ -536,6 +457,7 @@ mouseMoveWindow w = whenX (isClient w) $ withDisplay $ \d -> do
|
|||||||
(fromIntegral (fromIntegral (wa_y wa) + (ey - oy))))
|
(fromIntegral (fromIntegral (wa_y wa) + (ey - oy))))
|
||||||
(float w)
|
(float w)
|
||||||
|
|
||||||
|
-- | XXX comment me
|
||||||
mouseResizeWindow :: Window -> X ()
|
mouseResizeWindow :: Window -> X ()
|
||||||
mouseResizeWindow w = whenX (isClient w) $ withDisplay $ \d -> do
|
mouseResizeWindow w = whenX (isClient w) $ withDisplay $ \d -> do
|
||||||
io $ raiseWindow d w
|
io $ raiseWindow d w
|
||||||
@@ -544,8 +466,8 @@ mouseResizeWindow w = whenX (isClient w) $ withDisplay $ \d -> do
|
|||||||
io $ warpPointer d none w 0 0 0 0 (fromIntegral (wa_width wa)) (fromIntegral (wa_height wa))
|
io $ warpPointer d none w 0 0 0 0 (fromIntegral (wa_width wa)) (fromIntegral (wa_height wa))
|
||||||
mouseDrag (\ex ey -> do
|
mouseDrag (\ex ey -> do
|
||||||
io $ resizeWindow d w `uncurry`
|
io $ resizeWindow d w `uncurry`
|
||||||
applySizeHints sh ((fromIntegral (max 1 (ex - fromIntegral (wa_x wa)))),
|
applySizeHints sh (ex - fromIntegral (wa_x wa),
|
||||||
(fromIntegral (max 1 (ey - fromIntegral (wa_y wa))))))
|
ey - fromIntegral (wa_y wa)))
|
||||||
(float w)
|
(float w)
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
@@ -554,8 +476,13 @@ mouseResizeWindow w = whenX (isClient w) $ withDisplay $ \d -> do
|
|||||||
type D = (Dimension, Dimension)
|
type D = (Dimension, Dimension)
|
||||||
|
|
||||||
-- | Reduce the dimensions if needed to comply to the given SizeHints.
|
-- | Reduce the dimensions if needed to comply to the given SizeHints.
|
||||||
applySizeHints :: SizeHints -> D -> D
|
applySizeHints :: Integral a => SizeHints -> (a,a) -> D
|
||||||
applySizeHints sh =
|
applySizeHints sh (w,h) = applySizeHints' sh (fromIntegral $ max 1 w,
|
||||||
|
fromIntegral $ max 1 h)
|
||||||
|
|
||||||
|
-- | XXX comment me
|
||||||
|
applySizeHints' :: SizeHints -> D -> D
|
||||||
|
applySizeHints' sh =
|
||||||
maybe id applyMaxSizeHint (sh_max_size sh)
|
maybe id applyMaxSizeHint (sh_max_size sh)
|
||||||
. maybe id (\(bw, bh) (w, h) -> (w+bw, h+bh)) (sh_base_size sh)
|
. maybe id (\(bw, bh) (w, h) -> (w+bw, h+bh)) (sh_base_size sh)
|
||||||
. maybe id applyResizeIncHint (sh_resize_inc sh)
|
. maybe id applyResizeIncHint (sh_resize_inc sh)
|
@@ -1,43 +1,60 @@
|
|||||||
|
{-# LANGUAGE PatternGuards #-}
|
||||||
|
|
||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
-- |
|
-- |
|
||||||
-- Module : StackSet
|
-- Module : XMonad.StackSet
|
||||||
-- Copyright : (c) Don Stewart 2007
|
-- Copyright : (c) Don Stewart 2007
|
||||||
-- License : BSD3-style (see LICENSE)
|
-- License : BSD3-style (see LICENSE)
|
||||||
--
|
--
|
||||||
-- Maintainer : dons@cse.unsw.edu.au
|
-- Maintainer : dons@galois.com
|
||||||
-- Stability : experimental
|
-- Stability : experimental
|
||||||
-- Portability : portable, Haskell 98
|
-- Portability : portable, Haskell 98
|
||||||
--
|
--
|
||||||
|
|
||||||
module StackSet (
|
module XMonad.StackSet (
|
||||||
-- * Introduction
|
-- * Introduction
|
||||||
-- $intro
|
-- $intro
|
||||||
StackSet(..), Workspace(..), Screen(..), StackOrNot, Stack(..), RationalRect(..),
|
|
||||||
|
-- ** The Zipper
|
||||||
|
-- $zipper
|
||||||
|
|
||||||
|
-- ** Xinerama support
|
||||||
|
-- $xinerama
|
||||||
|
|
||||||
|
-- ** Master and Focus
|
||||||
|
-- $focus
|
||||||
|
|
||||||
|
StackSet(..), Workspace(..), Screen(..), Stack(..), RationalRect(..),
|
||||||
-- * Construction
|
-- * Construction
|
||||||
-- $construction
|
-- $construction
|
||||||
new, view, greedyView,
|
new, view, greedyView,
|
||||||
-- * Xinerama operations
|
-- * Xinerama operations
|
||||||
-- $xinerama
|
-- $xinerama
|
||||||
lookupWorkspace,
|
lookupWorkspace,
|
||||||
|
screens, workspaces, allWindows,
|
||||||
-- * Operations on the current stack
|
-- * Operations on the current stack
|
||||||
-- $stackOperations
|
-- $stackOperations
|
||||||
peek, index, integrate, integrate', differentiate,
|
peek, index, integrate, integrate', differentiate,
|
||||||
focusUp, focusDown,
|
focusUp, focusDown, focusMaster, focusWindow,
|
||||||
focusWindow, tagMember, member, findIndex,
|
tagMember, renameTag, ensureTags, member, findTag, mapWorkspace, mapLayout,
|
||||||
-- * Modifying the stackset
|
-- * Modifying the stackset
|
||||||
-- $modifyStackset
|
-- $modifyStackset
|
||||||
insertUp, delete, filter,
|
insertUp, delete, delete', filter,
|
||||||
-- * Setting the master window
|
-- * Setting the master window
|
||||||
-- $settingMW
|
-- $settingMW
|
||||||
swapMaster, swapUp, swapDown, modify, modify', float, sink, -- needed by users
|
swapUp, swapDown, swapMaster, modify, modify', float, sink, -- needed by users
|
||||||
-- * Composite operations
|
-- * Composite operations
|
||||||
-- $composite
|
-- $composite
|
||||||
shift
|
shift, shiftWin,
|
||||||
|
|
||||||
|
-- for testing
|
||||||
|
abort
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Prelude hiding (filter)
|
import Prelude hiding (filter)
|
||||||
import Data.Maybe (listToMaybe)
|
import Data.Maybe (listToMaybe,fromJust,isJust)
|
||||||
import qualified Data.List as L (delete,deleteBy,find,splitAt,filter)
|
import qualified Data.List as L (deleteBy,find,splitAt,filter,nub)
|
||||||
|
import Data.List ( (\\) )
|
||||||
import qualified Data.Map as M (Map,insert,delete,empty)
|
import qualified Data.Map as M (Map,insert,delete,empty)
|
||||||
|
|
||||||
-- $intro
|
-- $intro
|
||||||
@@ -48,18 +65,18 @@ import qualified Data.Map as M (Map,insert,delete,empty)
|
|||||||
-- window on each workspace has focus. The focused window on the current
|
-- window on each workspace has focus. The focused window on the current
|
||||||
-- workspace is the one which will take user input. It can be visualised
|
-- workspace is the one which will take user input. It can be visualised
|
||||||
-- as follows:
|
-- as follows:
|
||||||
--
|
--
|
||||||
-- > Workspace { 0*} { 1 } { 2 } { 3 } { 4 }
|
-- > Workspace { 0*} { 1 } { 2 } { 3 } { 4 }
|
||||||
-- >
|
-- >
|
||||||
-- > Windows [1 [] [3* [6*] []
|
-- > Windows [1 [] [3* [6*] []
|
||||||
-- > ,2*] ,4
|
-- > ,2*] ,4
|
||||||
-- > ,5]
|
-- > ,5]
|
||||||
--
|
--
|
||||||
-- Note that workspaces are indexed from 0, windows are numbered
|
-- Note that workspaces are indexed from 0, windows are numbered
|
||||||
-- uniquely. A '*' indicates the window on each workspace that has
|
-- uniquely. A '*' indicates the window on each workspace that has
|
||||||
-- focus, and which workspace is current.
|
-- focus, and which workspace is current.
|
||||||
--
|
|
||||||
-- Zipper
|
-- $zipper
|
||||||
--
|
--
|
||||||
-- We encode all the focus tracking directly in the data structure, with a 'zipper':
|
-- We encode all the focus tracking directly in the data structure, with a 'zipper':
|
||||||
--
|
--
|
||||||
@@ -70,7 +87,7 @@ import qualified Data.Map as M (Map,insert,delete,empty)
|
|||||||
-- The Zipper lets us replace an item deep in a complex data
|
-- The Zipper lets us replace an item deep in a complex data
|
||||||
-- structure, e.g., a tree or a term, without an mutation. The
|
-- structure, e.g., a tree or a term, without an mutation. The
|
||||||
-- resulting data structure will share as much of its components with
|
-- resulting data structure will share as much of its components with
|
||||||
-- the old structure as possible.
|
-- the old structure as possible.
|
||||||
--
|
--
|
||||||
-- Oleg Kiselyov, 27 Apr 2005, haskell\@, "Zipper as a delimited continuation"
|
-- Oleg Kiselyov, 27 Apr 2005, haskell\@, "Zipper as a delimited continuation"
|
||||||
--
|
--
|
||||||
@@ -87,25 +104,25 @@ import qualified Data.Map as M (Map,insert,delete,empty)
|
|||||||
-- Another good reference is:
|
-- Another good reference is:
|
||||||
--
|
--
|
||||||
-- The Zipper, Haskell wikibook
|
-- The Zipper, Haskell wikibook
|
||||||
--
|
|
||||||
-- Xinerama support:
|
-- $xinerama
|
||||||
--
|
|
||||||
-- Xinerama in X11 lets us view multiple virtual workspaces
|
-- Xinerama in X11 lets us view multiple virtual workspaces
|
||||||
-- simultaneously. While only one will ever be in focus (i.e. will
|
-- simultaneously. While only one will ever be in focus (i.e. will
|
||||||
-- receive keyboard events), other workspaces may be passively viewable.
|
-- receive keyboard events), other workspaces may be passively
|
||||||
-- We thus need to track which virtual workspaces are associated
|
-- viewable. We thus need to track which virtual workspaces are
|
||||||
-- (viewed) on which physical screens. We use a simple Map Workspace
|
-- associated (viewed) on which physical screens. To keep track of
|
||||||
-- Screen for this.
|
-- this, StackSet keeps separate lists of visible but non-focused
|
||||||
--
|
-- workspaces, and non-visible workspaces.
|
||||||
-- Master and Focus
|
|
||||||
|
-- $focus
|
||||||
--
|
--
|
||||||
-- Each stack tracks a focused item, and for tiling purposes also tracks
|
-- Each stack tracks a focused item, and for tiling purposes also tracks
|
||||||
-- a 'master' position. The connection between 'master' and 'focus'
|
-- a 'master' position. The connection between 'master' and 'focus'
|
||||||
-- needs to be well defined. Particular in relation to 'insert' and
|
-- needs to be well defined, particularly in relation to 'insert' and
|
||||||
-- 'delete'.
|
-- 'delete'.
|
||||||
--
|
--
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- API changes from xmonad 0.1:
|
-- API changes from xmonad 0.1:
|
||||||
-- StackSet constructor arguments changed. StackSet workspace window screen
|
-- StackSet constructor arguments changed. StackSet workspace window screen
|
||||||
--
|
--
|
||||||
@@ -117,19 +134,19 @@ import qualified Data.Map as M (Map,insert,delete,empty)
|
|||||||
--
|
--
|
||||||
-- * peek, -- was: peek\/peekStack
|
-- * peek, -- was: peek\/peekStack
|
||||||
--
|
--
|
||||||
-- * focusUp, focusDown, -- was: rotate
|
-- * focusUp, focusDown, -- was: rotate
|
||||||
--
|
--
|
||||||
-- * swapUp, swapDown
|
-- * swapUp, swapDown
|
||||||
--
|
--
|
||||||
-- * focus -- was: raiseFocus
|
-- * focus -- was: raiseFocus
|
||||||
--
|
--
|
||||||
-- * insertUp, -- was: insert\/push
|
-- * insertUp, -- was: insert\/push
|
||||||
--
|
--
|
||||||
-- * delete,
|
-- * delete,
|
||||||
--
|
--
|
||||||
-- * swapMaster, -- was: promote\/swap
|
-- * swapMaster, -- was: promote\/swap
|
||||||
--
|
--
|
||||||
-- * member,
|
-- * member,
|
||||||
--
|
--
|
||||||
-- * shift,
|
-- * shift,
|
||||||
--
|
--
|
||||||
@@ -139,32 +156,33 @@ import qualified Data.Map as M (Map,insert,delete,empty)
|
|||||||
--
|
--
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- |
|
-- |
|
||||||
-- A cursor into a non-empty list of workspaces.
|
-- A cursor into a non-empty list of workspaces.
|
||||||
--
|
--
|
||||||
-- We puncture the workspace list, producing a hole in the structure
|
-- We puncture the workspace list, producing a hole in the structure
|
||||||
-- used to track the currently focused workspace. The two other lists
|
-- used to track the currently focused workspace. The two other lists
|
||||||
-- that are produced are used to track those workspaces visible as
|
-- that are produced are used to track those workspaces visible as
|
||||||
-- Xinerama screens, and those workspaces not visible anywhere.
|
-- Xinerama screens, and those workspaces not visible anywhere.
|
||||||
|
|
||||||
data StackSet i a sid sd =
|
data StackSet i l a sid sd =
|
||||||
StackSet { current :: !(Screen i a sid sd) -- ^ currently focused workspace
|
StackSet { current :: !(Screen i l a sid sd) -- ^ currently focused workspace
|
||||||
, visible :: [Screen i a sid sd] -- ^ non-focused workspaces, visible in xinerama
|
, visible :: [Screen i l a sid sd] -- ^ non-focused workspaces, visible in xinerama
|
||||||
, hidden :: [Workspace i a] -- ^ workspaces not visible anywhere
|
, hidden :: [Workspace i l a] -- ^ workspaces not visible anywhere
|
||||||
, floating :: M.Map a RationalRect -- ^ floating windows
|
, floating :: M.Map a RationalRect -- ^ floating windows
|
||||||
} deriving (Show, Read, Eq)
|
} deriving (Show, Read, Eq)
|
||||||
|
|
||||||
-- | Visible workspaces, and their Xinerama screens.
|
-- | Visible workspaces, and their Xinerama screens.
|
||||||
data Screen i a sid sd = Screen { workspace :: !(Workspace i a)
|
data Screen i l a sid sd = Screen { workspace :: !(Workspace i l a)
|
||||||
, screen :: !sid
|
, screen :: !sid
|
||||||
, screenDetail :: !sd }
|
, screenDetail :: !sd }
|
||||||
deriving (Show, Read, Eq)
|
deriving (Show, Read, Eq)
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- A workspace is just a tag - its index - and a stack
|
-- A workspace is just a tag - its index - and a stack
|
||||||
--
|
--
|
||||||
data Workspace i a = Workspace { tag :: !i, stack :: StackOrNot a }
|
data Workspace i l a = Workspace { tag :: !i, layout :: l, stack :: Maybe (Stack a) }
|
||||||
deriving (Show, Read, Eq)
|
deriving (Show, Read, Eq)
|
||||||
|
|
||||||
|
-- | A structure for window geometries
|
||||||
data RationalRect = RationalRect Rational Rational Rational Rational
|
data RationalRect = RationalRect Rational Rational Rational Rational
|
||||||
deriving (Show, Read, Eq)
|
deriving (Show, Read, Eq)
|
||||||
|
|
||||||
@@ -186,8 +204,6 @@ data RationalRect = RationalRect Rational Rational Rational Rational
|
|||||||
-- structures, it is the differentiation of a [a], and integrating it
|
-- structures, it is the differentiation of a [a], and integrating it
|
||||||
-- back has a natural implementation used in 'index'.
|
-- back has a natural implementation used in 'index'.
|
||||||
--
|
--
|
||||||
type StackOrNot a = Maybe (Stack a)
|
|
||||||
|
|
||||||
data Stack a = Stack { focus :: !a -- focused thing in this set
|
data Stack a = Stack { focus :: !a -- focused thing in this set
|
||||||
, up :: [a] -- clowns to the left
|
, up :: [a] -- clowns to the left
|
||||||
, down :: [a] } -- jokers to the right
|
, down :: [a] } -- jokers to the right
|
||||||
@@ -201,48 +217,51 @@ abort x = error $ "xmonad: StackSet: " ++ x
|
|||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- $construction
|
-- $construction
|
||||||
|
|
||||||
-- | /O(n)/. Create a new stackset, of empty stacks, with given tags, with
|
-- | /O(n)/. Create a new stackset, of empty stacks, with given tags,
|
||||||
-- 'm' physical screens. 'm' should be less than or equal to the number of
|
-- with physical screens whose descriptions are given by 'm'. The
|
||||||
-- workspace tags. The first workspace in the list will be current.
|
-- number of physical screens (@length 'm'@) should be less than or
|
||||||
|
-- equal to the number of workspace tags. The first workspace in the
|
||||||
|
-- list will be current.
|
||||||
--
|
--
|
||||||
-- Xinerama: Virtual workspaces are assigned to physical screens, starting at 0.
|
-- Xinerama: Virtual workspaces are assigned to physical screens, starting at 0.
|
||||||
--
|
--
|
||||||
new :: (Integral s) => [i] -> [sd] -> StackSet i a s sd
|
new :: (Integral s) => l -> [i] -> [sd] -> StackSet i l a s sd
|
||||||
new wids m | not (null wids) && length m <= length wids = StackSet cur visi unseen M.empty
|
new l wids m | not (null wids) && length m <= length wids = StackSet cur visi unseen M.empty
|
||||||
where (seen,unseen) = L.splitAt (length m) $ map (flip Workspace Nothing) wids
|
where (seen,unseen) = L.splitAt (length m) $ map (\i -> Workspace i l Nothing) wids
|
||||||
(cur:visi) = [ Screen i s sd | (i, s, sd) <- zip3 seen [0..] m ]
|
(cur:visi) = [ Screen i s sd | (i, s, sd) <- zip3 seen [0..] m ]
|
||||||
-- now zip up visibles with their screen id
|
-- now zip up visibles with their screen id
|
||||||
new _ _ = abort "non-positive argument to StackSet.new"
|
new _ _ _ = abort "non-positive argument to StackSet.new"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- /O(w)/. Set focus to the workspace with index \'i\'.
|
-- /O(w)/. Set focus to the workspace with index \'i\'.
|
||||||
-- If the index is out of range, return the original StackSet.
|
-- If the index is out of range, return the original StackSet.
|
||||||
--
|
--
|
||||||
-- Xinerama: If the workspace is not visible on any Xinerama screen, it
|
-- Xinerama: If the workspace is not visible on any Xinerama screen, it
|
||||||
-- becomes the current screen. If it is in the visible list, it becomes
|
-- becomes the current screen. If it is in the visible list, it becomes
|
||||||
-- current.
|
-- current.
|
||||||
|
|
||||||
view :: (Eq a, Eq s, Eq i) => i -> StackSet i a s sd -> StackSet i a s sd
|
view :: (Eq s, Eq i) => i -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
view i s
|
view i s
|
||||||
| not (i `tagMember` s)
|
| not (i `tagMember` s)
|
||||||
|| i == tag (workspace (current s)) = s -- out of bounds or current
|
|| i == tag (workspace (current s)) = s -- out of bounds or current
|
||||||
|
|
||||||
| Just x <- L.find ((i==).tag.workspace) (visible s)
|
| Just x <- L.find ((i==).tag.workspace) (visible s)
|
||||||
-- if it is visible, it is just raised
|
-- if it is visible, it is just raised
|
||||||
= s { current = x, visible = current s : L.deleteBy screenEq x (visible s) }
|
= s { current = x, visible = current s : L.deleteBy (equating screen) x (visible s) }
|
||||||
|
|
||||||
| Just x <- L.find ((i==).tag) (hidden s)
|
| Just x <- L.find ((i==).tag) (hidden s) -- must be hidden then
|
||||||
-- if it was hidden, it is raised on the xine screen currently used
|
-- if it was hidden, it is raised on the xine screen currently used
|
||||||
= s { current = (current s) { workspace = x }
|
= s { current = (current s) { workspace = x }
|
||||||
, hidden = workspace (current s) : L.delete x (hidden s) }
|
, hidden = workspace (current s) : L.deleteBy (equating tag) x (hidden s) }
|
||||||
|
|
||||||
| otherwise = s
|
| otherwise = s -- can't happen: all workspaces are either invalid, current, visible, or hidden
|
||||||
where screenEq x y = screen x == screen y
|
|
||||||
|
where equating f = \x y -> f x == f y
|
||||||
|
|
||||||
-- 'Catch'ing this might be hard. Relies on monotonically increasing
|
-- 'Catch'ing this might be hard. Relies on monotonically increasing
|
||||||
-- workspace tags defined in 'new'
|
-- workspace tags defined in 'new'
|
||||||
|
--
|
||||||
|
-- and now tags are not monotonic, what happens here?
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- Set focus to the given workspace. If that workspace does not exist
|
-- Set focus to the given workspace. If that workspace does not exist
|
||||||
@@ -252,23 +271,22 @@ view i s
|
|||||||
-- screen, the workspaces of the current screen and the other screen are
|
-- screen, the workspaces of the current screen and the other screen are
|
||||||
-- swapped.
|
-- swapped.
|
||||||
|
|
||||||
greedyView :: (Eq a, Eq s, Eq i) => i -> StackSet i a s sd -> StackSet i a s sd
|
greedyView :: (Eq s, Eq i) => i -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
greedyView w ws
|
greedyView w ws
|
||||||
| any wTag (hidden ws) = view w ws
|
| any wTag (hidden ws) = view w ws
|
||||||
| (Just s) <- L.find (wTag . workspace) (visible ws)
|
| (Just s) <- L.find (wTag . workspace) (visible ws)
|
||||||
= ws { current = (current ws) { workspace = workspace s }
|
= ws { current = (current ws) { workspace = workspace s }
|
||||||
, visible = s { workspace = workspace (current ws) }
|
, visible = s { workspace = workspace (current ws) }
|
||||||
: L.filter (not . wTag . workspace) (visible ws) }
|
: L.filter (not . wTag . workspace) (visible ws) }
|
||||||
| otherwise = ws
|
| otherwise = ws
|
||||||
where
|
where wTag = (w == ) . tag
|
||||||
wTag = (w == ) . tag
|
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- $xinerama
|
-- $xinerama
|
||||||
|
|
||||||
-- | Find the tag of the workspace visible on Xinerama screen 'sc'.
|
-- | Find the tag of the workspace visible on Xinerama screen 'sc'.
|
||||||
-- Nothing if screen is out of bounds.
|
-- Nothing if screen is out of bounds.
|
||||||
lookupWorkspace :: Eq s => s -> StackSet i a s sd -> Maybe i
|
lookupWorkspace :: Eq s => s -> StackSet i l a s sd -> Maybe i
|
||||||
lookupWorkspace sc w = listToMaybe [ tag i | Screen i s _ <- current w : visible w, s == sc ]
|
lookupWorkspace sc w = listToMaybe [ tag i | Screen i s _ <- current w : visible w, s == sc ]
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
@@ -280,13 +298,13 @@ lookupWorkspace sc w = listToMaybe [ tag i | Screen i s _ <- current w : visible
|
|||||||
-- default value. Otherwise, it applies the function to the stack,
|
-- default value. Otherwise, it applies the function to the stack,
|
||||||
-- returning the result. It is like 'maybe' for the focused workspace.
|
-- returning the result. It is like 'maybe' for the focused workspace.
|
||||||
--
|
--
|
||||||
with :: b -> (Stack a -> b) -> StackSet i a s sd -> b
|
with :: b -> (Stack a -> b) -> StackSet i l a s sd -> b
|
||||||
with dflt f = maybe dflt f . stack . workspace . current
|
with dflt f = maybe dflt f . stack . workspace . current
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- Apply a function, and a default value for Nothing, to modify the current stack.
|
-- Apply a function, and a default value for Nothing, to modify the current stack.
|
||||||
--
|
--
|
||||||
modify :: StackOrNot a -> (Stack a -> StackOrNot a) -> StackSet i a s sd -> StackSet i a s sd
|
modify :: Maybe (Stack a) -> (Stack a -> Maybe (Stack a)) -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
modify d f s = s { current = (current s)
|
modify d f s = s { current = (current s)
|
||||||
{ workspace = (workspace (current s)) { stack = with d f s }}}
|
{ workspace = (workspace (current s)) { stack = with d f s }}}
|
||||||
|
|
||||||
@@ -294,14 +312,14 @@ modify d f s = s { current = (current s)
|
|||||||
-- Apply a function to modify the current stack if it isn't empty, and we don't
|
-- Apply a function to modify the current stack if it isn't empty, and we don't
|
||||||
-- want to empty it.
|
-- want to empty it.
|
||||||
--
|
--
|
||||||
modify' :: (Stack a -> Stack a) -> StackSet i a s sd -> StackSet i a s sd
|
modify' :: (Stack a -> Stack a) -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
modify' f = modify Nothing (Just . f)
|
modify' f = modify Nothing (Just . f)
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- /O(1)/. Extract the focused element of the current stack.
|
-- /O(1)/. Extract the focused element of the current stack.
|
||||||
-- Return Just that element, or Nothing for an empty stack.
|
-- Return Just that element, or Nothing for an empty stack.
|
||||||
--
|
--
|
||||||
peek :: StackSet i a s sd -> Maybe a
|
peek :: StackSet i l a s sd -> Maybe a
|
||||||
peek = with Nothing (return . focus)
|
peek = with Nothing (return . focus)
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
@@ -312,21 +330,22 @@ integrate (Stack x l r) = reverse l ++ x : r
|
|||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- /O(n)/ Flatten a possibly empty stack into a list.
|
-- /O(n)/ Flatten a possibly empty stack into a list.
|
||||||
integrate' :: StackOrNot a -> [a]
|
integrate' :: Maybe (Stack a) -> [a]
|
||||||
integrate' = maybe [] integrate
|
integrate' = maybe [] integrate
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- /O(n)/. Texture a list.
|
-- /O(n)/. Turn a list into a possibly empty stack (i.e., a zipper):
|
||||||
--
|
-- the first element of the list is current, and the rest of the list
|
||||||
differentiate :: [a] -> StackOrNot a
|
-- is down.
|
||||||
differentiate [] = Nothing
|
differentiate :: [a] -> Maybe (Stack a)
|
||||||
|
differentiate [] = Nothing
|
||||||
differentiate (x:xs) = Just $ Stack x [] xs
|
differentiate (x:xs) = Just $ Stack x [] xs
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- /O(n)/. 'filter p s' returns the elements of 's' such that 'p' evaluates to
|
-- /O(n)/. 'filter p s' returns the elements of 's' such that 'p' evaluates to
|
||||||
-- True. Order is preserved, and focus moves as described for 'delete'.
|
-- True. Order is preserved, and focus moves as described for 'delete'.
|
||||||
--
|
--
|
||||||
filter :: (a -> Bool) -> Stack a -> StackOrNot a
|
filter :: (a -> Bool) -> Stack a -> Maybe (Stack a)
|
||||||
filter p (Stack f ls rs) = case L.filter p (f:rs) of
|
filter p (Stack f ls rs) = case L.filter p (f:rs) of
|
||||||
f':rs' -> Just $ Stack f' (L.filter p ls) rs' -- maybe move focus down
|
f':rs' -> Just $ Stack f' (L.filter p ls) rs' -- maybe move focus down
|
||||||
[] -> case L.filter p ls of -- filter back up
|
[] -> case L.filter p ls of -- filter back up
|
||||||
@@ -339,24 +358,22 @@ filter p (Stack f ls rs) = case L.filter p (f:rs) of
|
|||||||
-- the head of the list. The implementation is given by the natural
|
-- the head of the list. The implementation is given by the natural
|
||||||
-- integration of a one-hole list cursor, back to a list.
|
-- integration of a one-hole list cursor, back to a list.
|
||||||
--
|
--
|
||||||
index :: StackSet i a s sd -> [a]
|
index :: StackSet i l a s sd -> [a]
|
||||||
index = with [] integrate
|
index = with [] integrate
|
||||||
|
|
||||||
-- let is = t : r ++ reverse l in take (length is) (dropWhile (/= m) (cycle is))
|
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- /O(1), O(w) on the wrapping case/.
|
-- /O(1), O(w) on the wrapping case/.
|
||||||
--
|
--
|
||||||
-- focusUp, focusDown. Move the window focus up or down the stack,
|
-- focusUp, focusDown. Move the window focus up or down the stack,
|
||||||
-- wrapping if we reach the end. The wrapping should model a -- 'cycle'
|
-- wrapping if we reach the end. The wrapping should model a 'cycle'
|
||||||
-- on the current stack. The 'master' window, and window order,
|
-- on the current stack. The 'master' window, and window order,
|
||||||
-- are unaffected by movement of focus.
|
-- are unaffected by movement of focus.
|
||||||
--
|
--
|
||||||
-- swapUp, swapDown, swap the neighbour in the stack ordering, wrapping
|
-- swapUp, swapDown, swap the neighbour in the stack ordering, wrapping
|
||||||
-- if we reach the end. Again the wrapping model should 'cycle' on
|
-- if we reach the end. Again the wrapping model should 'cycle' on
|
||||||
-- the current stack.
|
-- the current stack.
|
||||||
--
|
--
|
||||||
focusUp, focusDown, swapUp, swapDown :: StackSet i a s sd -> StackSet i a s sd
|
focusUp, focusDown, swapUp, swapDown :: StackSet i l a s sd -> StackSet i l a s sd
|
||||||
focusUp = modify' focusUp'
|
focusUp = modify' focusUp'
|
||||||
focusDown = modify' (reverseStack . focusUp' . reverseStack)
|
focusDown = modify' (reverseStack . focusUp' . reverseStack)
|
||||||
|
|
||||||
@@ -375,39 +392,69 @@ reverseStack :: Stack a -> Stack a
|
|||||||
reverseStack (Stack t ls rs) = Stack t rs ls
|
reverseStack (Stack t ls rs) = Stack t rs ls
|
||||||
|
|
||||||
--
|
--
|
||||||
-- | /O(1) on current window, O(n) in general/. Focus the window 'w',
|
-- | /O(1) on current window, O(n) in general/. Focus the window 'w',
|
||||||
-- and set its workspace as current.
|
-- and set its workspace as current.
|
||||||
--
|
--
|
||||||
focusWindow :: (Eq s, Eq a, Eq i) => a -> StackSet i a s sd -> StackSet i a s sd
|
focusWindow :: (Eq s, Eq a, Eq i) => a -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
focusWindow w s | Just w == peek s = s
|
focusWindow w s | Just w == peek s = s
|
||||||
| otherwise = maybe s id $ do
|
| otherwise = maybe s id $ do
|
||||||
n <- findIndex w s
|
n <- findTag w s
|
||||||
return $ until ((Just w ==) . peek) focusUp (view n s)
|
return $ until ((Just w ==) . peek) focusUp (view n s)
|
||||||
|
|
||||||
|
-- | Get a list of all screens in the StackSet.
|
||||||
|
screens :: StackSet i l a s sd -> [Screen i l a s sd]
|
||||||
|
screens s = current s : visible s
|
||||||
|
|
||||||
-- | Get a list of all workspaces in the StackSet.
|
-- | Get a list of all workspaces in the StackSet.
|
||||||
workspaces :: StackSet i a s sd -> [Workspace i a]
|
workspaces :: StackSet i l a s sd -> [Workspace i l a]
|
||||||
workspaces s = workspace (current s) : map workspace (visible s) ++ hidden s
|
workspaces s = workspace (current s) : map workspace (visible s) ++ hidden s
|
||||||
|
|
||||||
|
-- | Get a list of all windows in the StackSet in no particular order
|
||||||
|
allWindows :: Eq a => StackSet i l a s sd -> [a]
|
||||||
|
allWindows = L.nub . concatMap (integrate' . stack) . workspaces
|
||||||
|
|
||||||
-- | Is the given tag present in the StackSet?
|
-- | Is the given tag present in the StackSet?
|
||||||
tagMember :: Eq i => i -> StackSet i a s sd -> Bool
|
tagMember :: Eq i => i -> StackSet i l a s sd -> Bool
|
||||||
tagMember t = elem t . map tag . workspaces
|
tagMember t = elem t . map tag . workspaces
|
||||||
|
|
||||||
-- |
|
-- | Rename a given tag if present in the StackSet.
|
||||||
-- Finding if a window is in the stackset is a little tedious. We could
|
renameTag :: Eq i => i -> i -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
-- keep a cache :: Map a i, but with more bookkeeping.
|
renameTag o n = mapWorkspace rename
|
||||||
--
|
where rename w = if tag w == o then w { tag = n } else w
|
||||||
|
|
||||||
|
-- | Ensure that a given set of workspace tags is present by renaming
|
||||||
|
-- existing workspaces and\/or creating new hidden workspaces as
|
||||||
|
-- necessary.
|
||||||
|
ensureTags :: Eq i => l -> [i] -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
|
ensureTags l allt st = et allt (map tag (workspaces st) \\ allt) st
|
||||||
|
where et [] _ s = s
|
||||||
|
et (i:is) rn s | i `tagMember` s = et is rn s
|
||||||
|
et (i:is) [] s = et is [] (s { hidden = Workspace i l Nothing : hidden s })
|
||||||
|
et (i:is) (r:rs) s = et is rs $ renameTag r i s
|
||||||
|
|
||||||
|
-- | Map a function on all the workspaces in the StackSet.
|
||||||
|
mapWorkspace :: (Workspace i l a -> Workspace i l a) -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
|
mapWorkspace f s = s { current = updScr (current s)
|
||||||
|
, visible = map updScr (visible s)
|
||||||
|
, hidden = map f (hidden s) }
|
||||||
|
where updScr scr = scr { workspace = f (workspace scr) }
|
||||||
|
|
||||||
|
-- | Map a function on all the layouts in the StackSet.
|
||||||
|
mapLayout :: (l -> l') -> StackSet i l a s sd -> StackSet i l' a s sd
|
||||||
|
mapLayout f (StackSet v vs hs m) = StackSet (fScreen v) (map fScreen vs) (map fWorkspace hs) m
|
||||||
|
where
|
||||||
|
fScreen (Screen ws s sd) = Screen (fWorkspace ws) s sd
|
||||||
|
fWorkspace (Workspace t l s) = Workspace t (f l) s
|
||||||
|
|
||||||
-- | /O(n)/. Is a window in the StackSet.
|
-- | /O(n)/. Is a window in the StackSet.
|
||||||
member :: Eq a => a -> StackSet i a s sd -> Bool
|
member :: Eq a => a -> StackSet i l a s sd -> Bool
|
||||||
member a s = maybe False (const True) (findIndex a s)
|
member a s = isJust (findTag a s)
|
||||||
|
|
||||||
-- | /O(1) on current window, O(n) in general/.
|
-- | /O(1) on current window, O(n) in general/.
|
||||||
-- Return Just the workspace index of the given window, or Nothing
|
-- Return Just the workspace tag of the given window, or Nothing
|
||||||
-- if the window is not in the StackSet.
|
-- if the window is not in the StackSet.
|
||||||
findIndex :: Eq a => a -> StackSet i a s sd -> Maybe i
|
findTag :: Eq a => a -> StackSet i l a s sd -> Maybe i
|
||||||
findIndex a s = listToMaybe
|
findTag a s = listToMaybe
|
||||||
[ tag w | w <- workspaces s, has a (stack w) ]
|
[ tag w | w <- workspaces s, has a (stack w) ]
|
||||||
where has _ Nothing = False
|
where has _ Nothing = False
|
||||||
has x (Just (Stack t l r)) = x `elem` (t : l ++ r)
|
has x (Just (Stack t l r)) = x `elem` (t : l ++ r)
|
||||||
@@ -416,12 +463,10 @@ findIndex a s = listToMaybe
|
|||||||
-- $modifyStackset
|
-- $modifyStackset
|
||||||
|
|
||||||
-- |
|
-- |
|
||||||
-- /O(n)/. (Complexity due to duplicate check). Insert a new element into
|
-- /O(n)/. (Complexity due to duplicate check). Insert a new element
|
||||||
-- the stack, above the currently focused element.
|
-- into the stack, above the currently focused element. The new
|
||||||
--
|
-- element is given focus; the previously focused element is moved
|
||||||
-- The new element is given focus, and is set as the master window.
|
-- down.
|
||||||
-- The previously focused element is moved down. The previously
|
|
||||||
-- 'master' element is forgotten. (Thus, 'insert' will cause a retiling).
|
|
||||||
--
|
--
|
||||||
-- If the element is already in the stackset, the original stackset is
|
-- If the element is already in the stackset, the original stackset is
|
||||||
-- returned unmodified.
|
-- returned unmodified.
|
||||||
@@ -429,11 +474,11 @@ findIndex a s = listToMaybe
|
|||||||
-- Semantics in Huet's paper is that insert doesn't move the cursor.
|
-- Semantics in Huet's paper is that insert doesn't move the cursor.
|
||||||
-- However, we choose to insert above, and move the focus.
|
-- However, we choose to insert above, and move the focus.
|
||||||
--
|
--
|
||||||
insertUp :: Eq a => a -> StackSet i a s sd -> StackSet i a s sd
|
insertUp :: Eq a => a -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
insertUp a s = if member a s then s else insert
|
insertUp a s = if member a s then s else insert
|
||||||
where insert = modify (Just $ Stack a [] []) (\(Stack t l r) -> Just $ Stack a l (t:r)) s
|
where insert = modify (Just $ Stack a [] []) (\(Stack t l r) -> Just $ Stack a l (t:r)) s
|
||||||
|
|
||||||
-- insertDown :: a -> StackSet i a s sd -> StackSet i a s sd
|
-- insertDown :: a -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
-- insertDown a = modify (Stack a [] []) $ \(Stack t l r) -> Stack a (t:l) r
|
-- insertDown a = modify (Stack a [] []) $ \(Stack t l r) -> Stack a (t:l) r
|
||||||
-- Old semantics, from Huet.
|
-- Old semantics, from Huet.
|
||||||
-- > w { down = a : down w }
|
-- > w { down = a : down w }
|
||||||
@@ -452,22 +497,27 @@ insertUp a s = if member a s then s else insert
|
|||||||
-- * deleting the master window resets it to the newly focused window
|
-- * deleting the master window resets it to the newly focused window
|
||||||
-- * otherwise, delete doesn't affect the master.
|
-- * otherwise, delete doesn't affect the master.
|
||||||
--
|
--
|
||||||
delete :: (Ord a, Eq s) => a -> StackSet i a s sd -> StackSet i a s sd
|
delete :: (Ord a, Eq s) => a -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
delete w s = s { current = removeFromScreen (current s)
|
delete w = sink w . delete' w
|
||||||
, visible = map removeFromScreen (visible s)
|
|
||||||
, hidden = map removeFromWorkspace (hidden s) }
|
-- | Only temporarily remove the window from the stack, thereby not destroying special
|
||||||
|
-- information saved in the Stackset
|
||||||
|
delete' :: (Eq a, Eq s) => a -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
|
delete' w s = s { current = removeFromScreen (current s)
|
||||||
|
, visible = map removeFromScreen (visible s)
|
||||||
|
, hidden = map removeFromWorkspace (hidden s) }
|
||||||
where removeFromWorkspace ws = ws { stack = stack ws >>= filter (/=w) }
|
where removeFromWorkspace ws = ws { stack = stack ws >>= filter (/=w) }
|
||||||
removeFromScreen scr = scr { workspace = removeFromWorkspace (workspace scr) }
|
removeFromScreen scr = scr { workspace = removeFromWorkspace (workspace scr) }
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
-- | Given a window, and its preferred rectangle, set it as floating
|
-- | Given a window, and its preferred rectangle, set it as floating
|
||||||
-- A floating window should already be managed by the StackSet.
|
-- A floating window should already be managed by the StackSet.
|
||||||
float :: Ord a => a -> RationalRect -> StackSet i a s sd -> StackSet i a s sd
|
float :: Ord a => a -> RationalRect -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
float w r s = s { floating = M.insert w r (floating s) }
|
float w r s = s { floating = M.insert w r (floating s) }
|
||||||
|
|
||||||
-- | Clear the floating status of a window
|
-- | Clear the floating status of a window
|
||||||
sink :: Ord a => a -> StackSet i a s sd -> StackSet i a s sd
|
sink :: Ord a => a -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
sink w s = s { floating = M.delete w (floating s) }
|
sink w s = s { floating = M.delete w (floating s) }
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
@@ -476,24 +526,49 @@ sink w s = s { floating = M.delete w (floating s) }
|
|||||||
-- | /O(s)/. Set the master window to the focused window.
|
-- | /O(s)/. Set the master window to the focused window.
|
||||||
-- The old master window is swapped in the tiling order with the focused window.
|
-- The old master window is swapped in the tiling order with the focused window.
|
||||||
-- Focus stays with the item moved.
|
-- Focus stays with the item moved.
|
||||||
swapMaster :: StackSet i a s sd -> StackSet i a s sd
|
swapMaster :: StackSet i l a s sd -> StackSet i l a s sd
|
||||||
swapMaster = modify' $ \c -> case c of
|
swapMaster = modify' $ \c -> case c of
|
||||||
Stack _ [] _ -> c -- already master.
|
Stack _ [] _ -> c -- already master.
|
||||||
Stack t ls rs -> Stack t [] (ys ++ x : rs) where (x:ys) = reverse ls
|
Stack t ls rs -> Stack t [] (xs ++ x : rs) where (x:xs) = reverse ls
|
||||||
|
|
||||||
-- natural! keep focus, move current to the top, move top to current.
|
-- natural! keep focus, move current to the top, move top to current.
|
||||||
--
|
|
||||||
|
-- | /O(s)/. Set focus to the master window.
|
||||||
|
focusMaster :: StackSet i l a s sd -> StackSet i l a s sd
|
||||||
|
focusMaster = modify' $ \c -> case c of
|
||||||
|
Stack _ [] _ -> c
|
||||||
|
Stack t ls rs -> Stack x [] (xs ++ t : rs) where (x:xs) = reverse ls
|
||||||
|
|
||||||
|
--
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- $composite
|
-- $composite
|
||||||
|
|
||||||
-- | /O(w)/. shift. Move the focused element of the current stack to stack
|
-- | /O(w)/. shift. Move the focused element of the current stack to stack
|
||||||
-- 'n', leaving it as the focused element on that stack. The item is
|
-- 'n', leaving it as the focused element on that stack. The item is
|
||||||
-- inserted above the currently focused element on that workspace. --
|
-- inserted above the currently focused element on that workspace.
|
||||||
-- The actual focused workspace doesn't change. If there is -- no
|
-- The actual focused workspace doesn't change. If there is no
|
||||||
-- element on the current stack, the original stackSet is returned.
|
-- element on the current stack, the original stackSet is returned.
|
||||||
--
|
--
|
||||||
shift :: (Ord a, Eq s, Eq i) => i -> StackSet i a s sd -> StackSet i a s sd
|
shift :: (Ord a, Eq s, Eq i) => i -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
shift n s | n `tagMember` s && n /= curtag = maybe s go (peek s)
|
shift n s | n `tagMember` s && n /= curtag = maybe s go (peek s)
|
||||||
| otherwise = s
|
| otherwise = s
|
||||||
where go w = view curtag . insertUp w . view n . delete w $ s
|
where go w = view curtag . insertUp w . view n . delete' w $ s
|
||||||
curtag = tag (workspace (current s))
|
curtag = tag (workspace (current s))
|
||||||
|
|
||||||
|
-- | /O(n)/. shiftWin. Searches for the specified window 'w' on all workspaces
|
||||||
|
-- of the stackSet and moves it to stack 'n', leaving it as the focused
|
||||||
|
-- element on that stack. The item is inserted above the currently
|
||||||
|
-- focused element on that workspace.
|
||||||
|
-- The actual focused workspace doesn't change. If the window is not
|
||||||
|
-- found in the stackSet, the original stackSet is returned.
|
||||||
|
-- TODO how does this duplicate 'shift's behaviour?
|
||||||
|
shiftWin :: (Ord a, Eq a, Eq s, Eq i) => i -> a -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
|
shiftWin n w s | from == Nothing = s -- not found
|
||||||
|
| n `tagMember` s && (Just n) /= from = go
|
||||||
|
| otherwise = s
|
||||||
|
where from = findTag w s
|
||||||
|
|
||||||
|
go = on n (insertUp w) . on (fromJust from) (delete' w) $ s
|
||||||
|
curtag = tag (workspace (current s))
|
||||||
|
on i f = view curtag . f . view i
|
||||||
|
|
@@ -16,25 +16,19 @@ By utilising the expressivity of a modern functional language with a rich static
|
|||||||
.PP
|
.PP
|
||||||
You can switch to workspace N with mod-N. For example, to switch to workspace 5, you would press mod-5. Similarly, you can move the current window to another workspace with mod-shift-N.
|
You can switch to workspace N with mod-N. For example, to switch to workspace 5, you would press mod-5. Similarly, you can move the current window to another workspace with mod-shift-N.
|
||||||
.PP
|
.PP
|
||||||
When running with multiple monitors (Xinerama), each screen has exactly 1 workspace visible. When \fBxmonad\fR starts, workspace 1 is on screen 1, workspace 2 is on screen 2, etc. If you switch to a workspace which is currently visible on another screen, \fBxmonad\fR simply switches focus to that screen. If you switch to a workspace which is *not* visible, \fBxmonad\fR replaces the workspace on the *current* screen with the workspace you selected.
|
When running with multiple monitors (Xinerama), each screen has exactly 1 workspace visible. mod-{w,e,r} switch the focus between screens, while shift-mod-{w,e,r} move the current window to that screen. When \fBxmonad\fR starts, workspace 1 is on screen 1, workspace 2 is on screen 2, etc. When switching workspaces to one that is already visible, the current and visible workspaces are swapped.
|
||||||
.PP
|
.PP
|
||||||
For example, if you have the following configuration:
|
.SS Flags
|
||||||
.RS
|
\fBxmonad\fR has several flags which you may pass to the executable. These flags are:
|
||||||
.PP
|
.TP
|
||||||
Screen 1: Workspace 2
|
\fB--recompile
|
||||||
.PP
|
Recompiles your configuration in ~/.xmonad/xmonad.hs if it is newer than your already existing ~/.xmonad/xmonad executable.
|
||||||
Screen 2: Workspace 5 (current workspace)
|
.TP
|
||||||
.RE
|
\fB--recompile-force
|
||||||
.PP
|
Unconditionally recompiles your configuration in ~/.xmonad/xmonad.hs.
|
||||||
and you wanted to view workspace 7 on screen 1, you would press:
|
.TP
|
||||||
.RS
|
\fB--version
|
||||||
.PP
|
Display version of \fBxmonad\fR.
|
||||||
mod-2 (to select workspace 2, and make screen 1 the current screen)
|
|
||||||
.PP
|
|
||||||
mod-7 (to select workspace 7)
|
|
||||||
.RE
|
|
||||||
.PP
|
|
||||||
Since switching to the workspace currently visible on a given screen is such a common operation, shortcuts are provided: mod-{w,e,r} switch to the workspace currently visible on screens 1, 2, and 3 respectively. Likewise, shift-mod-{w,e,r} moves the current window to the workspace on that screen. Using these keys, the above example would become mod-w mod-7.
|
|
||||||
.SS Default keyboard bindings
|
.SS Default keyboard bindings
|
||||||
___KEYBINDINGS___
|
___KEYBINDINGS___
|
||||||
.SH EXAMPLES
|
.SH EXAMPLES
|
||||||
@@ -44,6 +38,6 @@ xmonad
|
|||||||
.RE
|
.RE
|
||||||
to your \fI~/.xinitrc\fR file
|
to your \fI~/.xinitrc\fR file
|
||||||
.SH CUSTOMIZATION
|
.SH CUSTOMIZATION
|
||||||
\fBxmonad\fR is customized by creating a custom Config.hs and (re)compiling the source code. After recompiling, 'restart' is used to fork the new version, with changes reflected immediately.
|
\fBxmonad\fR is customized in ~/.xmonad/xmonad.hs, and then restarting with mod-q.
|
||||||
.SH BUGS
|
.SH BUGS
|
||||||
Probably. If you find any, please report them: http://code.google.com/p/xmonad/issues/list
|
Probably. If you find any, please report them: http://code.google.com/p/xmonad/issues/list
|
||||||
|
283
man/xmonad.hs
Normal file
283
man/xmonad.hs
Normal file
@@ -0,0 +1,283 @@
|
|||||||
|
--
|
||||||
|
-- xmonad example config file.
|
||||||
|
--
|
||||||
|
-- A template showing all available configuration hooks,
|
||||||
|
-- and how to override the defaults in your own xmonad.hs conf file.
|
||||||
|
--
|
||||||
|
-- Normally, you'd only override those defaults you care about.
|
||||||
|
--
|
||||||
|
|
||||||
|
import XMonad
|
||||||
|
import System.Exit
|
||||||
|
|
||||||
|
import qualified XMonad.StackSet as W
|
||||||
|
import qualified Data.Map as M
|
||||||
|
|
||||||
|
-- The preferred terminal program, which is used in a binding below and by
|
||||||
|
-- certain contrib modules.
|
||||||
|
--
|
||||||
|
myTerminal = "xterm"
|
||||||
|
|
||||||
|
-- Width of the window border in pixels.
|
||||||
|
--
|
||||||
|
myBorderWidth = 1
|
||||||
|
|
||||||
|
-- modMask lets you specify which modkey you want to use. The default
|
||||||
|
-- is mod1Mask ("left alt"). You may also consider using mod3Mask
|
||||||
|
-- ("right alt"), which does not conflict with emacs keybindings. The
|
||||||
|
-- "windows key" is usually mod4Mask.
|
||||||
|
--
|
||||||
|
myModMask = mod1Mask
|
||||||
|
|
||||||
|
-- The mask for the numlock key. Numlock status is "masked" from the
|
||||||
|
-- current modifier status, so the keybindings will work with numlock on or
|
||||||
|
-- off. You may need to change this on some systems.
|
||||||
|
--
|
||||||
|
-- You can find the numlock modifier by running "xmodmap" and looking for a
|
||||||
|
-- modifier with Num_Lock bound to it:
|
||||||
|
--
|
||||||
|
-- > $ xmodmap | grep Num
|
||||||
|
-- > mod2 Num_Lock (0x4d)
|
||||||
|
--
|
||||||
|
-- Set numlockMask = 0 if you don't have a numlock key, or want to treat
|
||||||
|
-- numlock status separately.
|
||||||
|
--
|
||||||
|
myNumlockMask = mod2Mask
|
||||||
|
|
||||||
|
-- The default number of workspaces (virtual screens) and their names.
|
||||||
|
-- By default we use numeric strings, but any string may be used as a
|
||||||
|
-- workspace name. The number of workspaces is determined by the length
|
||||||
|
-- of this list.
|
||||||
|
--
|
||||||
|
-- A tagging example:
|
||||||
|
--
|
||||||
|
-- > workspaces = ["web", "irc", "code" ] ++ map show [4..9]
|
||||||
|
--
|
||||||
|
myWorkspaces = ["1","2","3","4","5","6","7","8","9"]
|
||||||
|
|
||||||
|
-- Border colors for unfocused and focused windows, respectively.
|
||||||
|
--
|
||||||
|
myNormalBorderColor = "#dddddd"
|
||||||
|
myFocusedBorderColor = "#ff0000"
|
||||||
|
|
||||||
|
-- Default offset of drawable screen boundaries from each physical
|
||||||
|
-- screen. Anything non-zero here will leave a gap of that many pixels
|
||||||
|
-- on the given edge, on the that screen. A useful gap at top of screen
|
||||||
|
-- for a menu bar (e.g. 15)
|
||||||
|
--
|
||||||
|
-- An example, to set a top gap on monitor 1, and a gap on the bottom of
|
||||||
|
-- monitor 2, you'd use a list of geometries like so:
|
||||||
|
--
|
||||||
|
-- > defaultGaps = [(18,0,0,0),(0,18,0,0)] -- 2 gaps on 2 monitors
|
||||||
|
--
|
||||||
|
-- Fields are: top, bottom, left, right.
|
||||||
|
--
|
||||||
|
myDefaultGaps = [(0,0,0,0)]
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Key bindings. Add, modify or remove key bindings here.
|
||||||
|
--
|
||||||
|
myKeys conf@(XConfig {XMonad.modMask = modMask}) = M.fromList $
|
||||||
|
|
||||||
|
-- launch a terminal
|
||||||
|
[ ((modMask .|. shiftMask, xK_Return), spawn $ XMonad.terminal conf)
|
||||||
|
|
||||||
|
-- launch dmenu
|
||||||
|
, ((modMask, xK_p ), spawn "exe=`dmenu_path | dmenu` && eval \"exec $exe\"")
|
||||||
|
|
||||||
|
-- launch gmrun
|
||||||
|
, ((modMask .|. shiftMask, xK_p ), spawn "gmrun")
|
||||||
|
|
||||||
|
-- close focused window
|
||||||
|
, ((modMask .|. shiftMask, xK_c ), kill)
|
||||||
|
|
||||||
|
-- Rotate through the available layout algorithms
|
||||||
|
, ((modMask, xK_space ), sendMessage NextLayout)
|
||||||
|
|
||||||
|
-- Reset the layouts on the current workspace to default
|
||||||
|
, ((modMask .|. shiftMask, xK_space ), setLayout $ XMonad.layoutHook conf)
|
||||||
|
|
||||||
|
-- Resize viewed windows to the correct size
|
||||||
|
, ((modMask, xK_n ), refresh)
|
||||||
|
|
||||||
|
-- Move focus to the next window
|
||||||
|
, ((modMask, xK_Tab ), windows W.focusDown)
|
||||||
|
|
||||||
|
-- Move focus to the next window
|
||||||
|
, ((modMask, xK_j ), windows W.focusDown)
|
||||||
|
|
||||||
|
-- Move focus to the previous window
|
||||||
|
, ((modMask, xK_k ), windows W.focusUp )
|
||||||
|
|
||||||
|
-- Move focus to the master window
|
||||||
|
, ((modMask, xK_m ), windows W.focusMaster )
|
||||||
|
|
||||||
|
-- Swap the focused window and the master window
|
||||||
|
, ((modMask, xK_Return), windows W.swapMaster)
|
||||||
|
|
||||||
|
-- Swap the focused window with the next window
|
||||||
|
, ((modMask .|. shiftMask, xK_j ), windows W.swapDown )
|
||||||
|
|
||||||
|
-- Swap the focused window with the previous window
|
||||||
|
, ((modMask .|. shiftMask, xK_k ), windows W.swapUp )
|
||||||
|
|
||||||
|
-- Shrink the master area
|
||||||
|
, ((modMask, xK_h ), sendMessage Shrink)
|
||||||
|
|
||||||
|
-- Expand the master area
|
||||||
|
, ((modMask, xK_l ), sendMessage Expand)
|
||||||
|
|
||||||
|
-- Push window back into tiling
|
||||||
|
, ((modMask, xK_t ), withFocused $ windows . W.sink)
|
||||||
|
|
||||||
|
-- Increment the number of windows in the master area
|
||||||
|
, ((modMask , xK_comma ), sendMessage (IncMasterN 1))
|
||||||
|
|
||||||
|
-- Deincrement the number of windows in the master area
|
||||||
|
, ((modMask , xK_period), sendMessage (IncMasterN (-1)))
|
||||||
|
|
||||||
|
-- toggle the status bar gap
|
||||||
|
, ((modMask , xK_b ),
|
||||||
|
modifyGap (\i n -> let x = (XMonad.defaultGaps conf ++ repeat (0,0,0,0)) !! i
|
||||||
|
in if n == x then (0,0,0,0) else x))
|
||||||
|
|
||||||
|
-- Quit xmonad
|
||||||
|
, ((modMask .|. shiftMask, xK_q ), io (exitWith ExitSuccess))
|
||||||
|
|
||||||
|
-- Restart xmonad
|
||||||
|
, ((modMask , xK_q ), restart "xmonad" True)
|
||||||
|
]
|
||||||
|
++
|
||||||
|
|
||||||
|
--
|
||||||
|
-- mod-[1..9], Switch to workspace N
|
||||||
|
-- mod-shift-[1..9], Move client to workspace N
|
||||||
|
--
|
||||||
|
[((m .|. modMask, k), windows $ f i)
|
||||||
|
| (i, k) <- zip (XMonad.workspaces conf) [xK_1 .. xK_9]
|
||||||
|
, (f, m) <- [(W.greedyView, 0), (W.shift, shiftMask)]]
|
||||||
|
++
|
||||||
|
|
||||||
|
--
|
||||||
|
-- mod-{w,e,r}, Switch to physical/Xinerama screens 1, 2, or 3
|
||||||
|
-- mod-shift-{w,e,r}, Move client to screen 1, 2, or 3
|
||||||
|
--
|
||||||
|
[((m .|. modMask, key), screenWorkspace sc >>= flip whenJust (windows . f))
|
||||||
|
| (key, sc) <- zip [xK_w, xK_e, xK_r] [0..]
|
||||||
|
, (f, m) <- [(W.view, 0), (W.shift, shiftMask)]]
|
||||||
|
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Mouse bindings: default actions bound to mouse events
|
||||||
|
--
|
||||||
|
myMouseBindings (XConfig {XMonad.modMask = modMask}) = M.fromList $
|
||||||
|
|
||||||
|
-- mod-button1, Set the window to floating mode and move by dragging
|
||||||
|
[ ((modMask, button1), (\w -> focus w >> mouseMoveWindow w))
|
||||||
|
|
||||||
|
-- mod-button2, Raise the window to the top of the stack
|
||||||
|
, ((modMask, button2), (\w -> focus w >> windows W.swapMaster))
|
||||||
|
|
||||||
|
-- mod-button3, Set the window to floating mode and resize by dragging
|
||||||
|
, ((modMask, button3), (\w -> focus w >> mouseResizeWindow w))
|
||||||
|
|
||||||
|
-- you may also bind events to the mouse scroll wheel (button4 and button5)
|
||||||
|
]
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Layouts:
|
||||||
|
|
||||||
|
-- You can specify and transform your layouts by modifying these values.
|
||||||
|
-- If you change layout bindings be sure to use 'mod-shift-space' after
|
||||||
|
-- restarting (with 'mod-q') to reset your layout state to the new
|
||||||
|
-- defaults, as xmonad preserves your old layout settings by default.
|
||||||
|
--
|
||||||
|
-- The available layouts. Note that each layout is separated by |||,
|
||||||
|
-- which denotes layout choice.
|
||||||
|
--
|
||||||
|
myLayout = tiled ||| Mirror tiled ||| Full
|
||||||
|
where
|
||||||
|
-- default tiling algorithm partitions the screen into two panes
|
||||||
|
tiled = Tall nmaster delta ratio
|
||||||
|
|
||||||
|
-- The default number of windows in the master pane
|
||||||
|
nmaster = 1
|
||||||
|
|
||||||
|
-- Default proportion of screen occupied by master pane
|
||||||
|
ratio = 1/2
|
||||||
|
|
||||||
|
-- Percent of screen to increment by when resizing panes
|
||||||
|
delta = 3/100
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Window rules:
|
||||||
|
|
||||||
|
-- Execute arbitrary actions and WindowSet manipulations when managing
|
||||||
|
-- a new window. You can use this to, for example, always float a
|
||||||
|
-- particular program, or have a client always appear on a particular
|
||||||
|
-- workspace.
|
||||||
|
--
|
||||||
|
-- To find the property name associated with a program, use
|
||||||
|
-- > xprop | grep WM_CLASS
|
||||||
|
-- and click on the client you're interested in.
|
||||||
|
--
|
||||||
|
-- To match on the WM_NAME, you can use 'title' in the same way that
|
||||||
|
-- 'className' and 'resource' are used below.
|
||||||
|
--
|
||||||
|
myManageHook = composeAll
|
||||||
|
[ className =? "MPlayer" --> doFloat
|
||||||
|
, className =? "Gimp" --> doFloat
|
||||||
|
, resource =? "desktop_window" --> doIgnore
|
||||||
|
, resource =? "kdesktop" --> doIgnore ]
|
||||||
|
|
||||||
|
-- Whether focus follows the mouse pointer.
|
||||||
|
myFocusFollowsMouse :: Bool
|
||||||
|
myFocusFollowsMouse = True
|
||||||
|
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Status bars and logging
|
||||||
|
|
||||||
|
-- Perform an arbitrary action on each internal state change or X event.
|
||||||
|
-- See the 'DynamicLog' extension for examples.
|
||||||
|
--
|
||||||
|
-- To emulate dwm's status bar
|
||||||
|
--
|
||||||
|
-- > logHook = dynamicLogDzen
|
||||||
|
--
|
||||||
|
myLogHook = return ()
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- Now run xmonad with all the defaults we set up.
|
||||||
|
|
||||||
|
-- Run xmonad with the settings you specify. No need to modify this.
|
||||||
|
--
|
||||||
|
main = xmonad defaults
|
||||||
|
|
||||||
|
-- A structure containing your configuration settings, overriding
|
||||||
|
-- fields in the default config. Any you don't override, will
|
||||||
|
-- use the defaults defined in xmonad/XMonad/Config.hs
|
||||||
|
--
|
||||||
|
-- No need to modify this.
|
||||||
|
--
|
||||||
|
defaults = defaultConfig {
|
||||||
|
-- simple stuff
|
||||||
|
terminal = myTerminal,
|
||||||
|
focusFollowsMouse = myFocusFollowsMouse,
|
||||||
|
borderWidth = myBorderWidth,
|
||||||
|
modMask = myModMask,
|
||||||
|
numlockMask = myNumlockMask,
|
||||||
|
workspaces = myWorkspaces,
|
||||||
|
normalBorderColor = myNormalBorderColor,
|
||||||
|
focusedBorderColor = myFocusedBorderColor,
|
||||||
|
defaultGaps = myDefaultGaps,
|
||||||
|
|
||||||
|
-- key bindings
|
||||||
|
keys = myKeys,
|
||||||
|
mouseBindings = myMouseBindings,
|
||||||
|
|
||||||
|
-- hooks, layouts
|
||||||
|
layoutHook = myLayout,
|
||||||
|
manageHook = myManageHook,
|
||||||
|
logHook = myLogHook
|
||||||
|
}
|
@@ -1,8 +1,8 @@
|
|||||||
{-# OPTIONS -fglasgow-exts #-}
|
{-# OPTIONS -fglasgow-exts -w #-}
|
||||||
|
module Properties where
|
||||||
|
|
||||||
import StackSet hiding (filter)
|
import XMonad.StackSet hiding (filter)
|
||||||
import qualified StackSet as S (filter)
|
import qualified XMonad.StackSet as S (filter)
|
||||||
import Operations (tile)
|
|
||||||
|
|
||||||
import Debug.Trace
|
import Debug.Trace
|
||||||
import Data.Word
|
import Data.Word
|
||||||
@@ -11,8 +11,10 @@ import Data.Ratio
|
|||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import System.Environment
|
import System.Environment
|
||||||
import Control.Exception (assert)
|
import Control.Exception (assert)
|
||||||
|
import qualified Control.Exception as C
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
import Test.QuickCheck hiding (promote)
|
import Test.QuickCheck hiding (promote)
|
||||||
|
import System.IO.Unsafe
|
||||||
import System.IO
|
import System.IO
|
||||||
import System.Random hiding (next)
|
import System.Random hiding (next)
|
||||||
import Text.Printf
|
import Text.Printf
|
||||||
@@ -28,18 +30,19 @@ import qualified Data.Map as M
|
|||||||
-- Some general hints for creating StackSet properties:
|
-- Some general hints for creating StackSet properties:
|
||||||
--
|
--
|
||||||
-- * ops that mutate the StackSet are usually local
|
-- * ops that mutate the StackSet are usually local
|
||||||
-- * most ops on StackSet should either be trivially reversible, or
|
-- * most ops on StackSet should either be trivially reversible, or
|
||||||
-- idempotent, or both.
|
-- idempotent, or both.
|
||||||
|
|
||||||
--
|
--
|
||||||
-- The all important Arbitrary instance for StackSet.
|
-- The all important Arbitrary instance for StackSet.
|
||||||
--
|
--
|
||||||
instance (Integral i, Integral s, Eq a, Arbitrary a, Arbitrary sd)
|
instance (Integral i, Integral s, Eq a, Arbitrary a, Arbitrary l, Arbitrary sd)
|
||||||
=> Arbitrary (StackSet i a s sd) where
|
=> Arbitrary (StackSet i l a s sd) where
|
||||||
arbitrary = do
|
arbitrary = do
|
||||||
sz <- choose (1,10) -- number of workspaces
|
sz <- choose (1,10) -- number of workspaces
|
||||||
n <- choose (0,sz-1) -- pick one to be in focus
|
n <- choose (0,sz-1) -- pick one to be in focus
|
||||||
sc <- choose (1,sz) -- a number of physical screens
|
sc <- choose (1,sz) -- a number of physical screens
|
||||||
|
lay <- arbitrary -- pick any layout
|
||||||
sds <- replicateM sc arbitrary
|
sds <- replicateM sc arbitrary
|
||||||
ls <- vector sz -- a vector of sz workspaces
|
ls <- vector sz -- a vector of sz workspaces
|
||||||
|
|
||||||
@@ -48,8 +51,7 @@ instance (Integral i, Integral s, Eq a, Arbitrary a, Arbitrary sd)
|
|||||||
else liftM Just (choose ((-1),length s-1))
|
else liftM Just (choose ((-1),length s-1))
|
||||||
| s <- ls ]
|
| s <- ls ]
|
||||||
|
|
||||||
return $ fromList (fromIntegral n, sds,fs,ls)
|
return $ fromList (fromIntegral n, sds,fs,ls,lay)
|
||||||
coarbitrary = error "no coarbitrary for StackSet"
|
|
||||||
|
|
||||||
|
|
||||||
-- | fromList. Build a new StackSet from a list of list of elements,
|
-- | fromList. Build a new StackSet from a list of list of elements,
|
||||||
@@ -62,14 +64,14 @@ instance (Integral i, Integral s, Eq a, Arbitrary a, Arbitrary sd)
|
|||||||
-- 'fs' random focused window on each workspace
|
-- 'fs' random focused window on each workspace
|
||||||
-- 'xs' list of list of windows
|
-- 'xs' list of list of windows
|
||||||
--
|
--
|
||||||
fromList :: (Integral i, Integral s, Eq a) => (i, [sd], [Maybe Int], [[a]]) -> StackSet i a s sd
|
fromList :: (Integral i, Integral s, Eq a) => (i, [sd], [Maybe Int], [[a]], l) -> StackSet i l a s sd
|
||||||
fromList (_,_,_,[]) = error "Cannot build a StackSet from an empty list"
|
fromList (_,_,_,[],_) = error "Cannot build a StackSet from an empty list"
|
||||||
|
|
||||||
fromList (o,m,fs,xs) =
|
fromList (o,m,fs,xs,l) =
|
||||||
let s = view o $
|
let s = view o $
|
||||||
foldr (\(i,ys) s ->
|
foldr (\(i,ys) s ->
|
||||||
foldr insertUp (view i s) ys)
|
foldr insertUp (view i s) ys)
|
||||||
(new [0..genericLength xs-1] m) (zip [0..] xs)
|
(new l [0..genericLength xs-1] m) (zip [0..] xs)
|
||||||
in foldr (\f t -> case f of
|
in foldr (\f t -> case f of
|
||||||
Nothing -> t
|
Nothing -> t
|
||||||
Just i -> foldr (const focusUp) t [0..i] ) s fs
|
Just i -> foldr (const focusUp) t [0..i] ) s fs
|
||||||
@@ -79,7 +81,7 @@ fromList (o,m,fs,xs) =
|
|||||||
--
|
--
|
||||||
-- Just generate StackSets with Char elements.
|
-- Just generate StackSets with Char elements.
|
||||||
--
|
--
|
||||||
type T = StackSet (NonNegative Int) Char Int Int
|
type T = StackSet (NonNegative Int) Int Char Int Int
|
||||||
|
|
||||||
-- Useful operation, the non-local workspaces
|
-- Useful operation, the non-local workspaces
|
||||||
hidden_spaces x = map workspace (visible x) ++ hidden x
|
hidden_spaces x = map workspace (visible x) ++ hidden x
|
||||||
@@ -87,7 +89,7 @@ hidden_spaces x = map workspace (visible x) ++ hidden x
|
|||||||
-- Basic data invariants of the StackSet
|
-- Basic data invariants of the StackSet
|
||||||
--
|
--
|
||||||
-- With the new zipper-based StackSet, tracking focus is no longer an
|
-- With the new zipper-based StackSet, tracking focus is no longer an
|
||||||
-- issue: the data structure enforces focus by construction.
|
-- issue: the data structure enforces focus by construction.
|
||||||
--
|
--
|
||||||
-- But we still need to ensure there are no duplicates, and master/and
|
-- But we still need to ensure there are no duplicates, and master/and
|
||||||
-- the xinerama mapping aren't checked by the data structure at all.
|
-- the xinerama mapping aren't checked by the data structure at all.
|
||||||
@@ -129,9 +131,9 @@ monotonic (x:y:zs) | x == y-1 = monotonic (y:zs)
|
|||||||
prop_invariant = invariant
|
prop_invariant = invariant
|
||||||
|
|
||||||
-- and check other ops preserve invariants
|
-- and check other ops preserve invariants
|
||||||
prop_empty_I (n :: Positive Int) = forAll (choose (1,fromIntegral n)) $ \m ->
|
prop_empty_I (n :: Positive Int) l = forAll (choose (1,fromIntegral n)) $ \m ->
|
||||||
forAll (vector m) $ \ms ->
|
forAll (vector m) $ \ms ->
|
||||||
invariant $ new [0..fromIntegral n-1] ms
|
invariant $ new l [0..fromIntegral n-1] ms
|
||||||
|
|
||||||
prop_view_I (n :: NonNegative Int) (x :: T) =
|
prop_view_I (n :: NonNegative Int) (x :: T) =
|
||||||
n `tagMember` x ==> invariant $ view (fromIntegral n) x
|
n `tagMember` x ==> invariant $ view (fromIntegral n) x
|
||||||
@@ -141,6 +143,8 @@ prop_greedyView_I (n :: NonNegative Int) (x :: T) =
|
|||||||
|
|
||||||
prop_focusUp_I (n :: NonNegative Int) (x :: T) =
|
prop_focusUp_I (n :: NonNegative Int) (x :: T) =
|
||||||
invariant $ foldr (const focusUp) x [1..n]
|
invariant $ foldr (const focusUp) x [1..n]
|
||||||
|
prop_focusMaster_I (n :: NonNegative Int) (x :: T) =
|
||||||
|
invariant $ foldr (const focusMaster) x [1..n]
|
||||||
prop_focusDown_I (n :: NonNegative Int) (x :: T) =
|
prop_focusDown_I (n :: NonNegative Int) (x :: T) =
|
||||||
invariant $ foldr (const focusDown) x [1..n]
|
invariant $ foldr (const focusDown) x [1..n]
|
||||||
|
|
||||||
@@ -167,21 +171,24 @@ prop_swap_right_I (n :: NonNegative Int) (x :: T) =
|
|||||||
prop_shift_I (n :: NonNegative Int) (x :: T) =
|
prop_shift_I (n :: NonNegative Int) (x :: T) =
|
||||||
n `tagMember` x ==> invariant $ shift (fromIntegral n) x
|
n `tagMember` x ==> invariant $ shift (fromIntegral n) x
|
||||||
|
|
||||||
|
prop_shift_win_I (n :: NonNegative Int) (w :: Char) (x :: T) =
|
||||||
|
n `tagMember` x && w `member` x ==> invariant $ shiftWin (fromIntegral n) w x
|
||||||
|
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- 'new'
|
-- 'new'
|
||||||
|
|
||||||
-- empty StackSets have no windows in them
|
-- empty StackSets have no windows in them
|
||||||
prop_empty (EmptyStackSet x) =
|
prop_empty (EmptyStackSet x) =
|
||||||
all (== Nothing) [ stack w | w <- workspace (current x)
|
all (== Nothing) [ stack w | w <- workspace (current x)
|
||||||
: map workspace (visible x) ++ hidden x ]
|
: map workspace (visible x) ++ hidden x ]
|
||||||
|
|
||||||
-- empty StackSets always have focus on first workspace
|
-- empty StackSets always have focus on first workspace
|
||||||
prop_empty_current (NonEmptyNubList ns) (NonEmptyNubList sds) =
|
prop_empty_current (NonEmptyNubList ns) (NonEmptyNubList sds) l =
|
||||||
-- TODO, this is ugly
|
-- TODO, this is ugly
|
||||||
length sds <= length ns ==>
|
length sds <= length ns ==>
|
||||||
tag (workspace $ current x) == head ns
|
tag (workspace $ current x) == head ns
|
||||||
where x = new ns sds :: T
|
where x = new l ns sds :: T
|
||||||
|
|
||||||
-- no windows will be a member of an empty workspace
|
-- no windows will be a member of an empty workspace
|
||||||
prop_member_empty i (EmptyStackSet x)
|
prop_member_empty i (EmptyStackSet x)
|
||||||
@@ -196,7 +203,7 @@ prop_view_current (x :: T) (n :: NonNegative Int) = i `tagMember` x ==>
|
|||||||
where
|
where
|
||||||
i = fromIntegral n
|
i = fromIntegral n
|
||||||
|
|
||||||
-- view *only* sets the current workspace, and touches Xinerama.
|
-- view *only* sets the current workspace, and touches Xinerama.
|
||||||
-- no workspace contents will be changed.
|
-- no workspace contents will be changed.
|
||||||
prop_view_local (x :: T) (n :: NonNegative Int) = i `tagMember` x ==>
|
prop_view_local (x :: T) (n :: NonNegative Int) = i `tagMember` x ==>
|
||||||
workspaces x == workspaces (view i x)
|
workspaces x == workspaces (view i x)
|
||||||
@@ -229,7 +236,7 @@ prop_greedyView_current (x :: T) (n :: NonNegative Int) = i `tagMember` x ==>
|
|||||||
where
|
where
|
||||||
i = fromIntegral n
|
i = fromIntegral n
|
||||||
|
|
||||||
-- greedyView *only* sets the current workspace, and touches Xinerama.
|
-- greedyView *only* sets the current workspace, and touches Xinerama.
|
||||||
-- no workspace contents will be changed.
|
-- no workspace contents will be changed.
|
||||||
prop_greedyView_local (x :: T) (n :: NonNegative Int) = i `tagMember` x ==>
|
prop_greedyView_local (x :: T) (n :: NonNegative Int) = i `tagMember` x ==>
|
||||||
workspaces x == workspaces (greedyView i x)
|
workspaces x == workspaces (greedyView i x)
|
||||||
@@ -287,13 +294,16 @@ prop_index_length (x :: T) =
|
|||||||
--
|
--
|
||||||
|
|
||||||
-- master/focus
|
-- master/focus
|
||||||
--
|
--
|
||||||
-- The tiling order, and master window, of a stack is unaffected by focus changes.
|
-- The tiling order, and master window, of a stack is unaffected by focus changes.
|
||||||
--
|
--
|
||||||
prop_focus_left_master (n :: NonNegative Int) (x::T) =
|
prop_focus_left_master (n :: NonNegative Int) (x::T) =
|
||||||
index (foldr (const focusUp) x [1..n]) == index x
|
index (foldr (const focusUp) x [1..n]) == index x
|
||||||
prop_focus_right_master (n :: NonNegative Int) (x::T) =
|
prop_focus_right_master (n :: NonNegative Int) (x::T) =
|
||||||
index (foldr (const focusDown) x [1..n]) == index x
|
index (foldr (const focusDown) x [1..n]) == index x
|
||||||
|
prop_focus_master_master (n :: NonNegative Int) (x::T) =
|
||||||
|
index (foldr (const focusMaster) x [1..n]) == index x
|
||||||
|
|
||||||
prop_focusWindow_master (n :: NonNegative Int) (x :: T) =
|
prop_focusWindow_master (n :: NonNegative Int) (x :: T) =
|
||||||
case peek x of
|
case peek x of
|
||||||
Nothing -> True
|
Nothing -> True
|
||||||
@@ -305,6 +315,9 @@ prop_focusWindow_master (n :: NonNegative Int) (x :: T) =
|
|||||||
prop_focus_left (x :: T) = (focusUp (focusDown x)) == x
|
prop_focus_left (x :: T) = (focusUp (focusDown x)) == x
|
||||||
prop_focus_right (x :: T) = (focusDown (focusUp x)) == x
|
prop_focus_right (x :: T) = (focusDown (focusUp x)) == x
|
||||||
|
|
||||||
|
-- focus master is idempotent
|
||||||
|
prop_focusMaster_idem (x :: T) = focusMaster x == focusMaster (focusMaster x)
|
||||||
|
|
||||||
-- focusWindow actually leaves the window focused...
|
-- focusWindow actually leaves the window focused...
|
||||||
prop_focusWindow_works (n :: NonNegative Int) (x :: T) =
|
prop_focusWindow_works (n :: NonNegative Int) (x :: T) =
|
||||||
case peek x of
|
case peek x of
|
||||||
@@ -323,7 +336,10 @@ prop_focus_all_r (x :: T) = (foldr (const focusDown) x [1..n]) == x
|
|||||||
-- f x' = foldr (\_ y -> rotate GT y) x' [1..n]
|
-- f x' = foldr (\_ y -> rotate GT y) x' [1..n]
|
||||||
|
|
||||||
-- focus is local to the current workspace
|
-- focus is local to the current workspace
|
||||||
prop_focus_local (x :: T) = hidden_spaces (focusDown x) == hidden_spaces x
|
prop_focus_down_local (x :: T) = hidden_spaces (focusDown x) == hidden_spaces x
|
||||||
|
prop_focus_up_local (x :: T) = hidden_spaces (focusUp x) == hidden_spaces x
|
||||||
|
|
||||||
|
prop_focus_master_local (x :: T) = hidden_spaces (focusMaster x) == hidden_spaces x
|
||||||
|
|
||||||
prop_focusWindow_local (n :: NonNegative Int) (x::T ) =
|
prop_focusWindow_local (n :: NonNegative Int) (x::T ) =
|
||||||
case peek x of
|
case peek x of
|
||||||
@@ -333,19 +349,21 @@ prop_focusWindow_local (n :: NonNegative Int) (x::T ) =
|
|||||||
in hidden_spaces (focusWindow (s !! i) x) == hidden_spaces x
|
in hidden_spaces (focusWindow (s !! i) x) == hidden_spaces x
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- member/findIndex
|
-- member/findTag
|
||||||
|
|
||||||
--
|
--
|
||||||
-- For all windows in the stackSet, findIndex should identify the
|
-- For all windows in the stackSet, findTag should identify the
|
||||||
-- correct workspace
|
-- correct workspace
|
||||||
--
|
--
|
||||||
prop_findIndex (x :: T) =
|
prop_findIndex (x :: T) =
|
||||||
and [ tag w == fromJust (findIndex i x)
|
and [ tag w == fromJust (findTag i x)
|
||||||
| w <- workspace (current x) : map workspace (visible x) ++ hidden x
|
| w <- workspace (current x) : map workspace (visible x) ++ hidden x
|
||||||
, t <- maybeToList (stack w)
|
, t <- maybeToList (stack w)
|
||||||
, i <- focus t : up t ++ down t
|
, i <- focus t : up t ++ down t
|
||||||
]
|
]
|
||||||
|
|
||||||
|
prop_allWindowsMember w (x :: T) = (w `elem` allWindows x) ==> member w x
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
-- 'insert'
|
-- 'insert'
|
||||||
|
|
||||||
@@ -438,7 +456,7 @@ prop_delete_focus_not_end (x :: T) =
|
|||||||
-- preserve order
|
-- preserve order
|
||||||
prop_filter_order (x :: T) =
|
prop_filter_order (x :: T) =
|
||||||
case stack $ workspace $ current x of
|
case stack $ workspace $ current x of
|
||||||
Nothing -> True
|
Nothing -> True
|
||||||
Just s@(Stack i _ _) -> integrate' (S.filter (/= i) s) == filter (/= i) (integrate' (Just s))
|
Just s@(Stack i _ _) -> integrate' (S.filter (/= i) s) == filter (/= i) (integrate' (Just s))
|
||||||
|
|
||||||
-- ---------------------------------------------------------------------
|
-- ---------------------------------------------------------------------
|
||||||
@@ -493,6 +511,113 @@ prop_shift_reversible i (x :: T) =
|
|||||||
y = swapMaster x
|
y = swapMaster x
|
||||||
n = tag (workspace $ current y)
|
n = tag (workspace $ current y)
|
||||||
|
|
||||||
|
-- ---------------------------------------------------------------------
|
||||||
|
-- shiftWin
|
||||||
|
|
||||||
|
-- shiftWin on current window is the same as shift
|
||||||
|
prop_shift_win_focus i (x :: T) =
|
||||||
|
i `tagMember` x ==> case peek x of
|
||||||
|
Nothing -> True
|
||||||
|
Just w -> shiftWin i w x == shift i x
|
||||||
|
|
||||||
|
-- shiftWin on a non-existant window is identity
|
||||||
|
prop_shift_win_indentity i w (x :: T) =
|
||||||
|
i `tagMember` x && not (w `member` x) ==> shiftWin i w x == x
|
||||||
|
|
||||||
|
-- shiftWin leaves the current screen as it is, if neither i is the tag
|
||||||
|
-- of the current workspace nor w on the current workspace
|
||||||
|
prop_shift_win_fix_current i w (x :: T) =
|
||||||
|
i `tagMember` x && w `member` x && i /= n && findTag w x /= Just n
|
||||||
|
==> (current $ x) == (current $ shiftWin i w x)
|
||||||
|
where
|
||||||
|
n = tag (workspace $ current x)
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- properties for the floating layer:
|
||||||
|
|
||||||
|
prop_float_reversible n (x :: T) =
|
||||||
|
n `member` x ==> sink n (float n geom x) == x
|
||||||
|
where
|
||||||
|
geom = RationalRect 100 100 100 100
|
||||||
|
|
||||||
|
-- check rectanges were set
|
||||||
|
{-
|
||||||
|
prop_float_sets_geometry n (x :: T) =
|
||||||
|
n `member` x ==> let y = float n geom x in M.lookup y (floating x) == Just geom
|
||||||
|
where
|
||||||
|
geom = RationalRect 100 100 100 100
|
||||||
|
-}
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
prop_screens (x :: T) = n `elem` screens x
|
||||||
|
where
|
||||||
|
n = current x
|
||||||
|
|
||||||
|
prop_differentiate xs =
|
||||||
|
if null xs then differentiate xs == Nothing
|
||||||
|
else (differentiate xs) == Just (Stack (head xs) [] (tail xs))
|
||||||
|
where _ = xs :: [Int]
|
||||||
|
|
||||||
|
-- looking up the tag of the current workspace should always produce a tag.
|
||||||
|
prop_lookup_current (x :: T) = lookupWorkspace scr x == Just tg
|
||||||
|
where
|
||||||
|
(Screen (Workspace tg _ _) scr _) = current x
|
||||||
|
|
||||||
|
-- looking at a visible tag
|
||||||
|
prop_lookup_visible (x :: T) =
|
||||||
|
visible x /= [] ==>
|
||||||
|
fromJust (lookupWorkspace scr x) `elem` tags
|
||||||
|
where
|
||||||
|
tags = [ tag (workspace y) | y <- visible x ]
|
||||||
|
scr = last [ screen y | y <- visible x ]
|
||||||
|
|
||||||
|
|
||||||
|
-- ---------------------------------------------------------------------
|
||||||
|
-- testing for failure
|
||||||
|
|
||||||
|
-- and help out hpc
|
||||||
|
prop_abort x = unsafePerformIO $ C.catch (abort "fail")
|
||||||
|
(\e -> return $ show e == "xmonad: StackSet: fail" )
|
||||||
|
where
|
||||||
|
_ = x :: Int
|
||||||
|
|
||||||
|
-- new should fail with an abort
|
||||||
|
prop_new_abort x = unsafePerformIO $ C.catch f
|
||||||
|
(\e -> return $ show e == "xmonad: StackSet: non-positive argument to StackSet.new" )
|
||||||
|
where
|
||||||
|
f = new undefined{-layout-} [] [] `seq` return False
|
||||||
|
|
||||||
|
_ = x :: Int
|
||||||
|
|
||||||
|
-- prop_view_should_fail = view {- with some bogus data -}
|
||||||
|
|
||||||
|
-- screens makes sense
|
||||||
|
prop_screens_works (x :: T) = screens x == current x : visible x
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
-- renaming tags
|
||||||
|
|
||||||
|
-- | Rename a given tag if present in the StackSet.
|
||||||
|
-- 408 renameTag :: Eq i => i -> i -> StackSet i l a s sd -> StackSet i l a s sd
|
||||||
|
|
||||||
|
prop_rename1 (x::T) o n = o `tagMember` x && not (n `tagMember` x) ==>
|
||||||
|
let y = renameTag o n x
|
||||||
|
in n `tagMember` y
|
||||||
|
|
||||||
|
prop_ensure (x :: T) l xs = let y = ensureTags l xs x
|
||||||
|
in and [ n `tagMember` y | n <- xs ]
|
||||||
|
|
||||||
|
prop_mapWorkspaceId (x::T) = x == mapWorkspace id x
|
||||||
|
|
||||||
|
prop_mapWorkspaceInverse (x::T) = x == mapWorkspace predTag (mapWorkspace succTag x)
|
||||||
|
where predTag w = w { tag = pred $ tag w }
|
||||||
|
succTag w = w { tag = succ $ tag w }
|
||||||
|
|
||||||
|
prop_mapLayoutId (x::T) = x == mapLayout id x
|
||||||
|
|
||||||
|
prop_mapLayoutInverse (x::T) = x == mapLayout pred (mapLayout succ x)
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- some properties for layouts:
|
-- some properties for layouts:
|
||||||
|
|
||||||
@@ -500,7 +625,7 @@ prop_shift_reversible i (x :: T) =
|
|||||||
{-
|
{-
|
||||||
prop_tile_fullscreen rect = tile pct rect 1 1 == [rect]
|
prop_tile_fullscreen rect = tile pct rect 1 1 == [rect]
|
||||||
|
|
||||||
-- multiple windows
|
-- multiple windows
|
||||||
prop_tile_non_overlap rect windows nmaster = noOverlaps (tile pct rect nmaster windows)
|
prop_tile_non_overlap rect windows nmaster = noOverlaps (tile pct rect nmaster windows)
|
||||||
where _ = rect :: Rectangle
|
where _ = rect :: Rectangle
|
||||||
|
|
||||||
@@ -526,7 +651,7 @@ noOverlaps xs = and [ verts a `notOverlap` verts b
|
|||||||
|
|
||||||
main :: IO ()
|
main :: IO ()
|
||||||
main = do
|
main = do
|
||||||
args <- getArgs
|
args <- fmap (drop 1) getArgs
|
||||||
let n = if null args then 100 else read (head args)
|
let n = if null args then 100 else read (head args)
|
||||||
(results, passed) <- liftM unzip $ mapM (\(s,a) -> printf "%-25s: " s >> a n) tests
|
(results, passed) <- liftM unzip $ mapM (\(s,a) -> printf "%-25s: " s >> a n) tests
|
||||||
printf "Passed %d tests!\n" (sum passed)
|
printf "Passed %d tests!\n" (sum passed)
|
||||||
@@ -561,20 +686,27 @@ main = do
|
|||||||
,("index/length" , mytest prop_index_length)
|
,("index/length" , mytest prop_index_length)
|
||||||
|
|
||||||
,("focus left : invariant", mytest prop_focusUp_I)
|
,("focus left : invariant", mytest prop_focusUp_I)
|
||||||
|
,("focus master : invariant", mytest prop_focusMaster_I)
|
||||||
,("focus right: invariant", mytest prop_focusDown_I)
|
,("focus right: invariant", mytest prop_focusDown_I)
|
||||||
,("focusWindow: invariant", mytest prop_focus_I)
|
,("focusWindow: invariant", mytest prop_focus_I)
|
||||||
,("focus left/master" , mytest prop_focus_left_master)
|
,("focus left/master" , mytest prop_focus_left_master)
|
||||||
,("focus right/master" , mytest prop_focus_right_master)
|
,("focus right/master" , mytest prop_focus_right_master)
|
||||||
|
,("focus master/master" , mytest prop_focus_master_master)
|
||||||
,("focusWindow master" , mytest prop_focusWindow_master)
|
,("focusWindow master" , mytest prop_focusWindow_master)
|
||||||
,("focus left/right" , mytest prop_focus_left)
|
,("focus left/right" , mytest prop_focus_left)
|
||||||
,("focus right/left" , mytest prop_focus_right)
|
,("focus right/left" , mytest prop_focus_right)
|
||||||
,("focus all left " , mytest prop_focus_all_l)
|
,("focus all left " , mytest prop_focus_all_l)
|
||||||
,("focus all right " , mytest prop_focus_all_r)
|
,("focus all right " , mytest prop_focus_all_r)
|
||||||
,("focus is local" , mytest prop_focus_local)
|
,("focus down is local" , mytest prop_focus_down_local)
|
||||||
|
,("focus up is local" , mytest prop_focus_up_local)
|
||||||
|
,("focus master is local" , mytest prop_focus_master_local)
|
||||||
|
,("focus master idemp" , mytest prop_focusMaster_idem)
|
||||||
|
|
||||||
,("focusWindow is local", mytest prop_focusWindow_local)
|
,("focusWindow is local", mytest prop_focusWindow_local)
|
||||||
,("focusWindow works" , mytest prop_focusWindow_works)
|
,("focusWindow works" , mytest prop_focusWindow_works)
|
||||||
|
|
||||||
,("findIndex" , mytest prop_findIndex)
|
,("findTag" , mytest prop_findIndex)
|
||||||
|
,("allWindows/member" , mytest prop_allWindowsMember)
|
||||||
|
|
||||||
,("insert: invariant" , mytest prop_insertUp_I)
|
,("insert: invariant" , mytest prop_insertUp_I)
|
||||||
,("insert/new" , mytest prop_insert_empty)
|
,("insert/new" , mytest prop_insert_empty)
|
||||||
@@ -611,6 +743,30 @@ main = do
|
|||||||
|
|
||||||
,("shift: invariant" , mytest prop_shift_I)
|
,("shift: invariant" , mytest prop_shift_I)
|
||||||
,("shift is reversible" , mytest prop_shift_reversible)
|
,("shift is reversible" , mytest prop_shift_reversible)
|
||||||
|
,("shiftWin: invariant" , mytest prop_shift_win_I)
|
||||||
|
,("shiftWin is shift on focus" , mytest prop_shift_win_focus)
|
||||||
|
,("shiftWin fix current" , mytest prop_shift_win_fix_current)
|
||||||
|
|
||||||
|
,("floating is reversible" , mytest prop_float_reversible)
|
||||||
|
,("screens includes current", mytest prop_screens)
|
||||||
|
,("differentiate works", mytest prop_differentiate)
|
||||||
|
,("lookupTagOnScreen", mytest prop_lookup_current)
|
||||||
|
,("lookupTagOnVisbleScreen", mytest prop_lookup_visible)
|
||||||
|
,("screens works", mytest prop_screens_works)
|
||||||
|
,("renaming works", mytest prop_rename1)
|
||||||
|
,("ensure works", mytest prop_ensure)
|
||||||
|
|
||||||
|
,("mapWorkspace id", mytest prop_mapWorkspaceId)
|
||||||
|
,("mapWorkspace inverse", mytest prop_mapWorkspaceInverse)
|
||||||
|
,("mapLayout id", mytest prop_mapLayoutId)
|
||||||
|
,("mapLayout inverse", mytest prop_mapLayoutInverse)
|
||||||
|
|
||||||
|
-- testing for failure:
|
||||||
|
,("abort fails", mytest prop_abort)
|
||||||
|
,("new fails with abort", mytest prop_new_abort)
|
||||||
|
,("shiftWin identity", mytest prop_shift_win_indentity)
|
||||||
|
|
||||||
|
-- renaming
|
||||||
|
|
||||||
{-
|
{-
|
||||||
,("tile 1 window fullsize", mytest prop_tile_fullscreen)
|
,("tile 1 window fullsize", mytest prop_tile_fullscreen)
|
||||||
@@ -781,8 +937,10 @@ instance Arbitrary EmptyStackSet where
|
|||||||
arbitrary = do
|
arbitrary = do
|
||||||
(NonEmptyNubList ns) <- arbitrary
|
(NonEmptyNubList ns) <- arbitrary
|
||||||
(NonEmptyNubList sds) <- arbitrary
|
(NonEmptyNubList sds) <- arbitrary
|
||||||
|
l <- arbitrary
|
||||||
-- there cannot be more screens than workspaces:
|
-- there cannot be more screens than workspaces:
|
||||||
return . EmptyStackSet . new ns $ take (min (length ns) (length sds)) sds
|
return . EmptyStackSet . new l ns $ take (min (length ns) (length sds)) sds
|
||||||
|
coarbitrary = error "coarbitrary EmptyStackSet"
|
||||||
|
|
||||||
-- | Generates a value that satisfies a predicate.
|
-- | Generates a value that satisfies a predicate.
|
||||||
suchThat :: Gen a -> (a -> Bool) -> Gen a
|
suchThat :: Gen a -> (a -> Bool) -> Gen a
|
||||||
|
@@ -9,7 +9,6 @@ main = do foo <- getContents
|
|||||||
-- uncomment the following to check for mistakes in isntcomment
|
-- uncomment the following to check for mistakes in isntcomment
|
||||||
-- putStr $ unlines $ actual_loc
|
-- putStr $ unlines $ actual_loc
|
||||||
|
|
||||||
isntcomment "" = False
|
|
||||||
isntcomment ('-':'-':_) = False
|
isntcomment ('-':'-':_) = False
|
||||||
isntcomment ('{':'-':_) = False -- pragmas
|
isntcomment ('{':'-':_) = False -- pragmas
|
||||||
isntcomment _ = True
|
isntcomment _ = True
|
||||||
|
@@ -4,14 +4,14 @@
|
|||||||
--
|
--
|
||||||
-- Format for the docstrings in Config.hs takes the following form:
|
-- Format for the docstrings in Config.hs takes the following form:
|
||||||
--
|
--
|
||||||
-- -- mod-x @@ Frob the whatsit
|
-- -- mod-x %! Frob the whatsit
|
||||||
--
|
--
|
||||||
-- "Frob the whatsit" will be used as the description for keybinding "mod-x"
|
-- "Frob the whatsit" will be used as the description for keybinding "mod-x"
|
||||||
--
|
--
|
||||||
-- If the keybinding name is omitted, it will try to guess from the rest of the
|
-- If the keybinding name is omitted, it will try to guess from the rest of the
|
||||||
-- line. For example:
|
-- line. For example:
|
||||||
--
|
--
|
||||||
-- [ ((modMask .|. shiftMask, xK_Return), spawn "xterm") -- @@ Launch an xterm
|
-- [ ((modMask .|. shiftMask, xK_Return), spawn "xterm") -- %! Launch an xterm
|
||||||
--
|
--
|
||||||
-- Here, mod-shift-return will be used as the keybinding name.
|
-- Here, mod-shift-return will be used as the keybinding name.
|
||||||
--
|
--
|
||||||
@@ -32,7 +32,7 @@ binding [ _, bindingLine, "", desc ] = (guessKeys bindingLine, desc)
|
|||||||
binding [ _, _, keyCombo, desc ] = (keyCombo, desc)
|
binding [ _, _, keyCombo, desc ] = (keyCombo, desc)
|
||||||
|
|
||||||
allBindings :: String -> [(String, String)]
|
allBindings :: String -> [(String, String)]
|
||||||
allBindings xs = map (binding . map trim) (xs =~ "(.*)--(.*)@@(.*)")
|
allBindings xs = map (binding . map trim) (xs =~ "(.*)--(.*)%!(.*)")
|
||||||
|
|
||||||
-- FIXME: What escaping should we be doing on these strings?
|
-- FIXME: What escaping should we be doing on these strings?
|
||||||
troff :: (String, String) -> String
|
troff :: (String, String) -> String
|
||||||
@@ -42,6 +42,6 @@ replace :: Eq a => a -> a -> [a] -> [a]
|
|||||||
replace x y = map (\a -> if a == x then y else a)
|
replace x y = map (\a -> if a == x then y else a)
|
||||||
|
|
||||||
main = do
|
main = do
|
||||||
troffBindings <- (concatMap troff . allBindings) `liftM` readFile "./Config.hs"
|
troffBindings <- (concatMap troff . allBindings) `liftM` readFile "./XMonad/Config.hs"
|
||||||
let sed = unlines . replace "___KEYBINDINGS___" troffBindings . lines
|
let sed = unlines . replace "___KEYBINDINGS___" troffBindings . lines
|
||||||
readFile "./man/xmonad.1.in" >>= return . sed >>= writeFile "./man/xmonad.1"
|
readFile "./man/xmonad.1.in" >>= return . sed >>= writeFile "./man/xmonad.1"
|
||||||
|
91
xmonad.cabal
91
xmonad.cabal
@@ -1,31 +1,76 @@
|
|||||||
name: xmonad
|
name: xmonad
|
||||||
version: 0.3
|
version: 0.6
|
||||||
homepage: http://xmonad.org
|
homepage: http://xmonad.org
|
||||||
synopsis: A lightweight X11 window manager.
|
synopsis: A tiling window manager
|
||||||
description:
|
description:
|
||||||
Xmonad is a minimalist tiling window manager for X, written in
|
xmonad is a tiling window manager for X. Windows are arranged
|
||||||
Haskell. Windows are managed using automatic layout algorithms,
|
automatically to tile the screen without gaps or overlap, maximising
|
||||||
which can be dynamically reconfigured. At any time windows are
|
screen use. All features of the window manager are accessible from
|
||||||
arranged so as to maximise the use of screen real estate. All
|
the keyboard: a mouse is strictly optional. xmonad is written and
|
||||||
features of the window manager are accessible purely from the
|
extensible in Haskell. Custom layout algorithms, and other
|
||||||
keyboard: a mouse is entirely optional. Xmonad is configured in
|
extensions, may be written by the user in config files. Layouts are
|
||||||
Haskell, and custom layout algorithms may be implemented by the user
|
applied dynamically, and different layouts may be used on each
|
||||||
in config files. A principle of Xmonad is predictability: the user
|
workspace. Xinerama is fully supported, allowing windows to be tiled
|
||||||
should know in advance precisely the window arrangement that will
|
on several screens.
|
||||||
result from any action.
|
|
||||||
category: System
|
category: System
|
||||||
license: BSD3
|
license: BSD3
|
||||||
license-file: LICENSE
|
license-file: LICENSE
|
||||||
author: Spencer Janssen
|
author: Spencer Janssen
|
||||||
maintainer: sjanssen@cse.unl.edu
|
maintainer: xmonad@haskell.org
|
||||||
build-depends: base>=2.0, X11>=1.2.1, X11-extras>=0.3, mtl>=1.0, unix>=1.0
|
extra-source-files: README TODO CONFIG STYLE tests/loc.hs tests/Properties.hs
|
||||||
extra-source-files: README TODO tests/loc.hs tests/Properties.hs man/xmonad.1.in
|
man/xmonad.1.in man/xmonad.1 man/xmonad.html man/xmonad.hs
|
||||||
Config.hs-boot util/GenerateManpage.hs man/xmonad.1 man/xmonad.html
|
util/GenerateManpage.hs
|
||||||
|
cabal-version: >= 1.2
|
||||||
|
|
||||||
executable: xmonad
|
flag small_base
|
||||||
main-is: Main.hs
|
description: Choose the new smaller, split-up base package.
|
||||||
other-modules: Config Operations StackSet XMonad
|
|
||||||
ghc-options: -funbox-strict-fields -O2 -fasm -Wall -optl-Wl,-s
|
flag testing
|
||||||
ghc-prof-options: -prof -auto-all
|
description: Testing mode, only build minimal components
|
||||||
extensions: GeneralizedNewtypeDeriving
|
default: False
|
||||||
-- Also requires deriving Typeable
|
|
||||||
|
library
|
||||||
|
exposed-modules: XMonad
|
||||||
|
XMonad.Main
|
||||||
|
XMonad.Core
|
||||||
|
XMonad.Config
|
||||||
|
XMonad.Layout
|
||||||
|
XMonad.ManageHook
|
||||||
|
XMonad.Operations
|
||||||
|
XMonad.StackSet
|
||||||
|
|
||||||
|
if flag(small_base)
|
||||||
|
build-depends: base >= 3, containers, directory, process
|
||||||
|
else
|
||||||
|
build-depends: base < 3
|
||||||
|
build-depends: X11>=1.4.1, mtl, unix
|
||||||
|
|
||||||
|
ghc-options: -funbox-strict-fields -Wall -optl-Wl,-s
|
||||||
|
ghc-prof-options: -prof -auto-all
|
||||||
|
extensions: CPP
|
||||||
|
|
||||||
|
if flag(testing)
|
||||||
|
buildable: False
|
||||||
|
|
||||||
|
executable xmonad
|
||||||
|
main-is: Main.hs
|
||||||
|
other-modules: XMonad
|
||||||
|
XMonad.Main
|
||||||
|
XMonad.Core
|
||||||
|
XMonad.Config
|
||||||
|
XMonad.Layout
|
||||||
|
XMonad.ManageHook
|
||||||
|
XMonad.Operations
|
||||||
|
XMonad.StackSet
|
||||||
|
|
||||||
|
ghc-options: -funbox-strict-fields -Wall -optl-Wl,-s
|
||||||
|
ghc-prof-options: -prof -auto-all
|
||||||
|
extensions: CPP
|
||||||
|
|
||||||
|
if flag(testing)
|
||||||
|
cpp-options: -DTESTING
|
||||||
|
hs-source-dirs: . tests/
|
||||||
|
build-depends: QuickCheck < 2
|
||||||
|
ghc-options: -Werror
|
||||||
|
if flag(testing) && flag(small_base)
|
||||||
|
build-depends: random
|
||||||
|
Reference in New Issue
Block a user