aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--ManPrompt.hs52
1 files changed, 27 insertions, 25 deletions
diff --git a/ManPrompt.hs b/ManPrompt.hs
index ad0b2e7..2eddb39 100644
--- a/ManPrompt.hs
+++ b/ManPrompt.hs
@@ -1,3 +1,4 @@
+{-# OPTIONS_GHC -Wall #-}
-----------------------------------------------------------------------------
-- |
-- Module : XMonadContrib.ManPrompt
@@ -6,7 +7,7 @@
--
-- Maintainer : valery.vv@gmail.com
-- Stability : unstable
--- Portability : unportable
+-- Portability : non-portable (uses \"manpath\" and \"bash\")
--
-- A manual page prompt for XMonad window manager.
--
@@ -15,9 +16,7 @@
-- * narrow completions by section number, if the one is specified
-- (like @\/etc\/bash_completion@ does)
--
--- * handle explicit paths (e.g., @~\/src\/xmonad\/man\/xmonad.1@)
---
--- * quickcheck properties
+-- * test with QuickCheck
-----------------------------------------------------------------------------
module XMonadContrib.ManPrompt (
@@ -48,7 +47,7 @@ import Data.Maybe
--
-- 2. In your keybindings add something like:
--
--- > , ((modMask, xK_F1), manPrompt defaultXPConfig)
+-- > , ((modMask, xK_F1), manPrompt defaultXPConfig) -- mod-f1 %! Query for manual page to be displayed
-- %import XMonadContrib.XPrompt
-- %import XMonadContrib.ManPrompt
@@ -64,33 +63,36 @@ manPrompt :: XPConfig -> X ()
manPrompt c = mkXPrompt Man c manCompl $ runInTerm . (++) "man "
manCompl :: String -> IO [String]
-manCompl s = getManpages >>= flip mkComplFunFromList s
-
--- | Obtain the list of manual pages.
---
--- /XXX Code duplication!/
--- Adopted from 'ShellPrompt.getCommands'.
-getManpages :: IO [String]
-getManpages = do
- p <- getCommandOutput "manpath -g 2>/dev/null" `E.catch` const (return [])
- let sections = ["man" ++ show n | n <- [1..9 :: Int]] -- XXX "cat1".."cat9"?
- ds = [d ++ "/" ++ s | d <- split ':' p, s <- sections]
- stripSec = reverse . drop 1 . dropWhile (/= '.') . reverse
- ms <- forM ds $ \d -> do
- exists <- doesDirectoryExist d
- if exists
- then map (stripSec . stripSuffixes [".gz", ".bz2"]) `fmap`
- getDirectoryContents d
- else return []
- return . uniqSort . concat $ ms
+manCompl str | '/' `elem` str = do
+ -- XXX It may be better to use readline instead of bash's compgen...
+ lines `fmap` getCommandOutput ("bash -c 'compgen -A file " ++ str ++ "'")
+ | otherwise = do
+ mp <- getCommandOutput "manpath -g 2>/dev/null" `E.catch` \_ -> return []
+ let sects = ["man" ++ show n | n <- [1..9 :: Int]]
+ dirs = [d ++ "/" ++ s | d <- split ':' mp, s <- sects]
+ stripExt = reverse . drop 1 . dropWhile (/= '.') . reverse
+ mans <- forM dirs $ \d -> do
+ exists <- doesDirectoryExist d
+ if exists
+ then map (stripExt . stripSuffixes [".gz", ".bz2"]) `fmap`
+ getDirectoryContents d
+ else return []
+ mkComplFunFromList (uniqSort $ concat mans) str
-- | Run a command using shell and return its output.
+--
+-- XXX merge with 'Run.runProcessWithInput'?
+--
+-- * update documentation of the latter (there is no 'Maybe' in result)
+--
+-- * ask \"gurus\" whether @evaluate (length ...)@ approach is
+-- better\/more idiomatic
getCommandOutput :: String -> IO String
getCommandOutput s = do
(pin, pout, perr, ph) <- runInteractiveCommand s
hClose pin
output <- hGetContents pout
- E.evaluate (null output)
+ E.evaluate (length output)
hClose perr
waitForProcess ph
return output