aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--XMonad/Doc/Developing.hs202
1 files changed, 97 insertions, 105 deletions
diff --git a/XMonad/Doc/Developing.hs b/XMonad/Doc/Developing.hs
index 180fc8a..ced3175 100644
--- a/XMonad/Doc/Developing.hs
+++ b/XMonad/Doc/Developing.hs
@@ -8,29 +8,29 @@
-- Stability : unstable
-- Portability : portable
--
--- This module documents the xmonad internals.
+-- This module documents the xmonad internals. It is intended for
+-- advanced users who are curious about the xmonad source code and
+-- want an brief overview. This document may also be helpful for the
+-- beginner\/intermediate Haskell programmer who is motivated to write
+-- an xmonad extension as a way to deepen her understanding of this
+-- powerful functional language; however, there is not space here to
+-- go into much detail. A more comprehensive document introducing
+-- beginner\/intermediate Haskell programmers to the xmonad source is
+-- planned for the xmonad users' wiki
+-- (<http://haskell.org/haskellwiki/Xmonad>).
--
--- It is intended for the advanced users who are curious about the
--- xmonad source code and want an brief overview of it.
---
--- While some knowledge of Haskell is required, still this document is
--- also intended for the beginner\/intermediate Haskell programmer who
--- could find writing an xmonad extension a motivation for deepening
--- her understanding of this powerful functional language.
---
--- It may be useful also for those who would like to extend xmonad. If
--- you think your extension may be useful for other users too, you may
--- consider about releasing it.
---
--- Coding guidelines and licencing policies must be followed if you
--- want your code to be included in the official repositories.
+-- If you write an extension module and think it may be useful for
+-- others, consider releasing it. Coding guidelines and licensing
+-- policies are covered at the end of this document, and must be
+-- followed if you want your code to be included in the official
+-- repositories.
--
-----------------------------------------------------------------------------
module XMonad.Doc.Developing
(
- -- -- * Writing new extensions
- -- -- $writing
+ -- * Writing new extensions
+ -- $writing
-- * Libraries for writing window managers
-- $xmonad-libs
@@ -68,30 +68,23 @@ module XMonad.Doc.Developing
{- $xmonad-libs
-Starting from version 0.5, xmonad and xmonad-contrib are packaged and
-distributed as libraries. This way of distributing xmonad has many
-advantages, since it allows the packaging by GNU\/Linux distributions
-while letting the user have the possibility of greatly customizing the
-window manager to fit her needs.
+Starting with version 0.5, xmonad and xmonad-contrib are packaged and
+distributed as libraries, instead of components which must be compiled
+by the user into a binary (as they were prior to version 0.5). This
+way of distributing xmonad has many advantages, since it allows
+packaging by GNU\/Linux distributions while still allowing the user to
+customize the window manager to fit her needs.
Basically, xmonad and the xmonad-contrib libraries let users write
-their own window manager in a matter of a few lines of code.
+their own window manager in just a few lines of code. While
+@~\/.xmonad\/xmonad.hs@ at first seems to be simply a configuration
+file, it is actually a complete Haskell program which uses the xmonad
+and xmonad-contrib libraries to create a custom window manager.
-In fact, what seems to be just a configuration file,
-@~\/.xmonad\/xmonad.hs@ (whose presence is not necessary for running
-the default configuration), is indeed a full Haskell program, with its
-@main@ entry point.
-
-This makes it possible, not only to edit the xmonad default
-configuration, as we have seen the "XMonad.Doc.Extending" document,
-but also to use the Haskell programming language to extend the tasks
-performed by the window manager you are writing every time you write
-your own @~\/.xmonad\/xmonad.hs@.
-
-This is obviously out of the scope of this document, which instead
-will be focused on the xmonad internals, by describing, very briefly,
-the programming techniques that have been employed in order to make
-writing new extensions very simple.
+This makes it possible not only to edit the default xmonad
+configuration, as we have seen in the "XMonad.Doc.Extending" document,
+but to use the Haskell programming language to extend the window
+manager you are writing in any way you see fit.
-}
@@ -106,21 +99,20 @@ Xsession starting script. This binary, whose code can be read in
@Main.hs@ of the xmonad source tree, will use 'XMonad.Core.recompile'
to run @ghc@ in order to build a binary from @~\/.xmonad\/xmonad.hs@.
If this compilation process fails, for any reason, a default @main@
-entry point will be used, which calls 'XMonad.Main.xmonad', from the
-"XMonad.Main" module.
-
-So, the real @main@ entry point, the one that even users' application
-in @~\/.xmonad\/xmonad.hs@ must call, is 'XMonad.Main.xmonad'
+entry point will be used, which calls the 'XMonad.Main.xmonad'
+function with a default configuration.
-'XMonad.Main.xmonad' takes the configuration as its only argument,
-configuration whose type ('XMonad.Core.XConfig') is defined in
-"XMonad.Core".
+Thus, the real @main@ entry point, the one that even the users' custom
+window manager application in @~\/.xmonad\/xmonad.hs@ must call, is
+the 'XMonad.Main.xmonad' function. This function takes a configuration
+as its only argument, whose type ('XMonad.Core.XConfig')
+is defined in "XMonad.Core".
'XMonad.Main.xmonad' takes care of opening the connection with the X
server, initializing the state (or deserializing it when restarted)
and the configuration, and calling the event handler
-('XMonad.Main.handle') that will 'Prelude.forever' loop waiting for
-events and acting accordingly.
+('XMonad.Main.handle') that goes into an infinite loop (using
+'Prelude.forever') waiting for events and acting accordingly.
-}
@@ -130,25 +122,26 @@ The event loop which calls 'XMonad.Main.handle' to react to events is
run within the 'XMonad.Core.X' monad, which is a
'Control.Monad.State.StateT' transformer over 'IO', encapsulated
within a 'Control.Monad.Reader.ReaderT' transformer. The
-'Control.Monad.State.StateT' transformer encapsulates the (writable)
-state of the window manager ('XMonad.Core.XState'), whereas the
-'Control.Monad.Reader.ReaderT' transformer encapsulates the
-(read-only) configuration ('XMonad.Core.XConf').
+'Control.Monad.State.StateT' transformer encapsulates the
+(read\/writable) state of the window manager (of type
+'XMonad.Core.XState'), whereas the 'Control.Monad.Reader.ReaderT'
+transformer encapsulates the (read-only) configuration (of type
+'XMonad.Core.XConf').
-Thanks to the @newtype deriving@ clause the instance of the
+Thanks to GHC's newtype deriving feature, the instance of the
'Control.Monad.State.MonadState' class parametrized over
'XMonad.Core.XState' and the instance of the
'Control.Monad.Reader.MonadReader' class parametrized over
-'XMonad.Core.XConf' are automatically derived from us by ghc. This way
-we can use 'Control.Monad.State.get', 'Control.Monad.State.gets' and
-'Control.Monad.State.modify' for the 'XMonad.Core.XState', and
-'Control.Monad.Reader.ask' and 'Control.Monad.Reader.asks' for
-reading the 'XMonad.Core.XConf'.
+'XMonad.Core.XConf' are automatically derived for the 'XMonad.Core.X'
+monad. This way we can use 'Control.Monad.State.get',
+'Control.Monad.State.gets' and 'Control.Monad.State.modify' for the
+'XMonad.Core.XState', and 'Control.Monad.Reader.ask' and
+'Control.Monad.Reader.asks' for reading the 'XMonad.Core.XConf'.
'XMonad.Core.XState' is where all the sensitive information about
-windows managing is stored. And the main record of the
+window management is stored. The most important field of the
'XMonad.Core.XState' is the 'XMonad.Core.windowset', whose type
-('XMonad.Core.WindowSet') is a type synonymous for a
+('XMonad.Core.WindowSet') is a synonym for a
'XMonad.StackSet.StackSet' parametrized over a
'XMonad.Core.WorkspaceID' (a 'String'), a layout type wrapped inside
the 'XMonad.Layout.Layout' existential data type, the
@@ -156,81 +149,80 @@ the 'XMonad.Layout.Layout' existential data type, the
'XMonad.Core.ScreenDetail's.
What a 'XMonad.StackSet.StackSet' is and how it can be manipulated
-with pure functions is perfectly described in the Haddock
-documentation of the "XMonad.StackSet" module, and will not be repeated
-here.
+with pure functions is described in the Haddock documentation of the
+"XMonad.StackSet" module.
-The 'XMonad.StackSet.StackSet' ('XMonad.Core.WindowSet') has 4
-records:
+The 'XMonad.StackSet.StackSet' ('XMonad.Core.WindowSet') has four
+fields:
* 'XMonad.StackSet.current', for the current, focused workspace. This
- is a 'XMonad.StackSet.Screen', composed by a
- 'XMonad.StackSet.Workspace', and the screen information (for
+ is a 'XMonad.StackSet.Screen', which is composed of a
+ 'XMonad.StackSet.Workspace' together with the screen information (for
Xinerama support).
* 'XMonad.StackSet.visible', a list of 'XMonad.StackSet.Screen's for
- the other visible (with Xinerama) workspaces.
+ the other visible (with Xinerama) workspaces. For non-Xinerama
+ setups, this list is always empty.
-* 'XMonad.StackSet.hidden', the list of 'XMonad.StackSet.Screen's for
- non visible workspaces.
+* 'XMonad.StackSet.hidden', the list of non-visible
+ 'XMonad.StackSet.Workspace's.
+
+* 'XMonad.StackSet.floating', a map from floating
+ 'Graphics.X11.Types.Window's to 'XMonad.StackSet.RationalRect's
+ specifying their geometry.
The 'XMonad.StackSet.Workspace' type is made of a
'XMonad.StackSet.tag', a 'XMonad.StackSet.layout' and
-'XMonad.StackSet.stack', possibly empy, of windows.
-
-"XMonad.StackSet", to be imported qualified, provides many pure
-functions to manipulate the 'XMonad.StackSet.StackSet'. These
-functions are usually used as the argument of
-'XMonad.Operations.windows', which indeed takes a pure function to
-manipulate the 'XMonad.Core.WindowSet' and does all the needed
-operations to refresh the screen and save the modified
+a (possibly empty) 'XMonad.StackSet.stack' of windows.
+
+"XMonad.StackSet" (which should usually be imported qualified, to
+avoid name clashes with Prelude functions such as 'Prelude.delete' and
+'Prelude.filter') provides many pure functions to manipulate the
+'XMonad.StackSet.StackSet'. These functions are most commonlyq used as
+an argument to 'XMonad.Operations.windows', which takes a pure
+function to manipulate the 'XMonad.Core.WindowSet' and does all the
+needed operations to refresh the screen and save the modified
'XMonad.Core.XState'.
-During 'XMonad.Operations.windows' calls the 'XMonad.StackSet.layout'
-record of the 'XMonad.StackSet.current' and 'XMonad.StackSet.visible'
-'XMonad.StackSet.Workspace's is used to arrange the
-'XMonad.StackSet.stack' of windows of each workspace.
+During each 'XMonad.Operations.windows' call, the
+'XMonad.StackSet.layout' field of the 'XMonad.StackSet.current' and
+'XMonad.StackSet.visible' 'XMonad.StackSet.Workspace's are used to
+physically arrange the 'XMonad.StackSet.stack' of windows on each
+workspace.
The possibility of manipulating the 'XMonad.StackSet.StackSet'
('XMonad.Core.WindowSet') with pure functions makes it possible to
test all the properties of those functions with QuickCheck, providing
-greater reliability of the core code.
-
-Every change to the "XMonad.StackSet" module must be accompanied with
-the set of property to be tested with QuickCheck before being applied.
+greater reliability of the core code. Every change to the
+"XMonad.StackSet" module must be accompanied by appropriate QuickCheck
+properties before being applied.
-}
{- $events
-Events and event handling are the main data and activity xmonad is
-involved with. And X Events are one of the most important.
-
-Still there may be events that are generated by layouts, or by the
-user, for sending commands to layouts, for instance.
+Event handling is the core activity of xmonad. Events generated by
+the X server are most important, but there may also be events
+generated by layouts or the user.
"XMonad.Core" defines a class that generalizes the concept of events,
'XMonad.Core.Message', constrained to types with a
'Data.Typeable.Typeable' instance definition (which can be
-automatically derived by ghc).
-
-'XMonad.Core.Message's are wrapped within an existential type
-'XMonad.Core.SomeMessage'.
-
-The 'Data.Typeable.Typeable' constraint allows us to define a
+automatically derived by ghc). 'XMonad.Core.Message's are wrapped
+within an existential type 'XMonad.Core.SomeMessage'. The
+'Data.Typeable.Typeable' constraint allows for the definition of a
'XMonad.Core.fromMessage' function that can unwrap the message with
-'Data.Typeable.cast'.
-
-X Events are instances of this class.
+'Data.Typeable.cast'. X Events are instances of this class, along
+with any messages used by xmonad itself or by extension modules.
-By using the 'Data.Typeable.Typeable' class for any kind of
-'XMonad.Core.Message's and events we can define polymorphic functions
-and use them for processing messages or unhandled events.
+Using the 'Data.Typeable.Typeable' class for any kind of
+'XMonad.Core.Message's and events allows us to define polymorphic functions
+for processing messages or unhandled events.
This is precisely what happens with X events: xmonad passes them to
'XMonad.Main.handle'. If the main event handling function doesn't have
anything to do with the event, the event is sent to all visible
-layouts by 'XMonad.Operations.bradcastMessage'.
+layouts by 'XMonad.Operations.broadcastMessage'.
This messaging system allows the user to create new message types,
simply declare an instance of the 'Data.Typeable.Typeable' and use
@@ -261,7 +253,7 @@ xmonad contributed extensions.
* 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'
+ crash, so do not call 'error' or 'undefined'.
* Tabs are /illegal/. Use 4 spaces for indenting.