aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--XMonad/Actions/SpawnOn.hs77
-rw-r--r--XMonad/Hooks/ManageHelpers.hs12
-rw-r--r--XMonad/Prompt/Shell.hs5
-rw-r--r--xmonad-contrib.cabal1
4 files changed, 93 insertions, 2 deletions
diff --git a/XMonad/Actions/SpawnOn.hs b/XMonad/Actions/SpawnOn.hs
new file mode 100644
index 0000000..d452cdf
--- /dev/null
+++ b/XMonad/Actions/SpawnOn.hs
@@ -0,0 +1,77 @@
+-----------------------------------------------------------------------------
+-- |
+-- Module : XMonad.Actions.SpawnOn
+-- Copyright : (c) Spencer Janssen
+-- License : BSD
+--
+-- Maintainer : Spencer Janssen <spencerjanssen@gmail.com>
+-- Stability : unstable
+-- Portability : unportable
+--
+-- This module provides helper functions to be used in @manageHook@. Here's
+-- how you might use this:
+--
+-- > import XMonad.Hooks.ManageHelpers
+-- > main = do
+-- > sp <- mkSpawner
+-- > xmonad defaultConfig {
+-- > ...
+-- > manageHook = spawnHook sp <+> manageHook defaultConfig
+-- > ...
+-- > }
+
+module XMonad.Actions.SpawnOn (
+ Spawner,
+ mkSpawner,
+ manageSpawn,
+ spawnHere,
+ spawnOn,
+ shellPromptHere,
+ shellPromptOn
+) where
+
+import Data.IORef
+import System.Posix.Types (ProcessID)
+
+import XMonad
+import qualified XMonad.StackSet as W
+
+import XMonad.Hooks.ManageHelpers
+import XMonad.Prompt
+import XMonad.Prompt.Shell
+
+newtype Spawner = Spawner {pidsRef :: IORef [(ProcessID, WorkspaceId)]}
+
+maxPids :: Int
+maxPids = 5
+
+mkSpawner :: (Functor m, MonadIO m) => m Spawner
+mkSpawner = io . fmap Spawner $ newIORef []
+
+manageSpawn :: Spawner -> ManageHook
+manageSpawn sp = do
+ pids <- io . readIORef $ pidsRef sp
+ mp <- pid
+ case flip lookup pids =<< mp of
+ Just w -> doF (W.shift w)
+ Nothing -> doF id
+
+mkPrompt :: (String -> X ()) -> XPConfig -> X ()
+mkPrompt cb c = do
+ cmds <- io $ getCommands
+ mkXPrompt Shell c (getShellCompl cmds) cb
+
+shellPromptHere :: Spawner -> XPConfig -> X ()
+shellPromptHere sp = mkPrompt (spawnHere sp)
+
+shellPromptOn :: Spawner -> WorkspaceId -> XPConfig -> X ()
+shellPromptOn sp ws = mkPrompt (spawnOn sp ws)
+
+spawnHere :: Spawner -> String -> X ()
+spawnHere sp cmd = withWindowSet $ \ws -> spawnOn sp (currTag ws) cmd
+ where currTag = W.tag . W.workspace . W.current
+
+spawnOn :: Spawner -> WorkspaceId -> String -> X ()
+spawnOn sp ws cmd = do
+ p <- spawnPID cmd
+ io $ modifyIORef (pidsRef sp) (take maxPids . ((p, ws) :))
diff --git a/XMonad/Hooks/ManageHelpers.hs b/XMonad/Hooks/ManageHelpers.hs
index 5dd8e32..e5f25e4 100644
--- a/XMonad/Hooks/ManageHelpers.hs
+++ b/XMonad/Hooks/ManageHelpers.hs
@@ -31,6 +31,7 @@ module XMonad.Hooks.ManageHelpers (
isKDETrayWindow,
isFullscreen,
isDialog,
+ pid,
transientTo,
maybeToDefinite,
MaybeManageHook,
@@ -49,6 +50,8 @@ import qualified XMonad.StackSet as W
import Data.Maybe
import Data.Monoid
+import System.Posix (ProcessID)
+
-- | Denotes a side of a screen. @S@ stands for South, @NE@ for Northwest
-- etc. @C@ stands for Center.
data Side = SC | NC | CE | CW | SE | SW | NE | NW | C
@@ -144,6 +147,15 @@ isDialog = ask >>= \w -> liftX $ do
Just xs -> fromIntegral w_dialog `elem` xs
_ -> False
+pid :: Query (Maybe ProcessID)
+pid = ask >>= \w -> liftX $ do
+ dpy <- asks display
+ a <- getAtom "_NET_WM_PID"
+ p <- io $ getWindowProperty32 dpy a w
+ return $ case p of
+ Just [x] -> Just (fromIntegral x)
+ _ -> Nothing
+
-- | A predicate to check whether a window is Transient.
-- It holds the result which might be the window it is transient to
-- or it might be 'Nothing'.
diff --git a/XMonad/Prompt/Shell.hs b/XMonad/Prompt/Shell.hs
index c659787..d1ea150 100644
--- a/XMonad/Prompt/Shell.hs
+++ b/XMonad/Prompt/Shell.hs
@@ -15,7 +15,8 @@
module XMonad.Prompt.Shell
( -- * Usage
-- $usage
- shellPrompt
+ Shell (..)
+ , shellPrompt
, getCommands
, getBrowser
, getEditor
@@ -141,4 +142,4 @@ getBrowser = env "BROWSER" "firefox"
-- | Like 'getBrowser', but should be of a text editor. This gets the $EDITOR variable, defaulting to \"emacs\".
getEditor :: IO String
-getEditor = env "EDITOR" "emacs" \ No newline at end of file
+getEditor = env "EDITOR" "emacs"
diff --git a/xmonad-contrib.cabal b/xmonad-contrib.cabal
index f1d93a0..81c87fa 100644
--- a/xmonad-contrib.cabal
+++ b/xmonad-contrib.cabal
@@ -95,6 +95,7 @@ library
XMonad.Actions.Search
XMonad.Actions.SimpleDate
XMonad.Actions.SinkAll
+ XMonad.Actions.SpawnOn
XMonad.Actions.Submap
XMonad.Actions.SwapWorkspaces
XMonad.Actions.TagWindows