127 lines
		
	
	
	
		
			3.6 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
			
		
		
	
	
			127 lines
		
	
	
	
		
			3.6 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
{- Freedesktop.org specifications
 | 
						|
 -
 | 
						|
 - http://standards.freedesktop.org/basedir-spec/latest/
 | 
						|
 - http://standards.freedesktop.org/desktop-entry-spec/latest/
 | 
						|
 - http://standards.freedesktop.org/menu-spec/latest/
 | 
						|
 -
 | 
						|
 - Copyright 2012 Joey Hess <joey@kitenet.net>
 | 
						|
 -
 | 
						|
 - Licensed under the GNU GPL version 3 or higher.
 | 
						|
 -}
 | 
						|
 | 
						|
module Utility.FreeDesktop (
 | 
						|
	DesktopEntry,	
 | 
						|
	genDesktopEntry,
 | 
						|
	buildDesktopMenuFile,
 | 
						|
	writeDesktopMenuFile,
 | 
						|
	desktopMenuFilePath,
 | 
						|
	autoStartPath,
 | 
						|
	systemDataDir,
 | 
						|
	systemConfigDir,
 | 
						|
	userDataDir,
 | 
						|
	userConfigDir,
 | 
						|
	userDesktopDir
 | 
						|
) where
 | 
						|
 | 
						|
import Utility.Exception
 | 
						|
import Utility.Path
 | 
						|
import Utility.UserInfo
 | 
						|
import Utility.Process
 | 
						|
import Utility.PartialPrelude
 | 
						|
 | 
						|
import System.Environment
 | 
						|
import System.Directory
 | 
						|
import System.FilePath
 | 
						|
import Data.List
 | 
						|
import Data.String.Utils
 | 
						|
import Control.Applicative
 | 
						|
 | 
						|
type DesktopEntry = [(Key, Value)]
 | 
						|
 | 
						|
type Key = String
 | 
						|
 | 
						|
data Value = StringV String | BoolV Bool | NumericV Float | ListV [Value]
 | 
						|
 | 
						|
toString :: Value -> String
 | 
						|
toString (StringV s) = s
 | 
						|
toString (BoolV b)
 | 
						|
	| b = "true"
 | 
						|
	| otherwise = "false"
 | 
						|
toString(NumericV f) = show f
 | 
						|
toString (ListV l)
 | 
						|
	| null l = ""
 | 
						|
	| otherwise = (intercalate ";" $ map (escapesemi . toString) l) ++ ";"
 | 
						|
  where
 | 
						|
	escapesemi = join "\\;" . split ";"
 | 
						|
 | 
						|
genDesktopEntry :: String -> String -> Bool -> FilePath -> [String] -> DesktopEntry
 | 
						|
genDesktopEntry name comment terminal program categories =
 | 
						|
	[ item "Type" StringV "Application"
 | 
						|
	, item "Version" NumericV 1.0
 | 
						|
	, item "Name" StringV name
 | 
						|
	, item "Comment" StringV comment
 | 
						|
	, item "Terminal" BoolV terminal
 | 
						|
	, item "Exec" StringV program
 | 
						|
	, item "Categories" ListV (map StringV categories)
 | 
						|
	]
 | 
						|
  where
 | 
						|
	item x c y = (x, c y)
 | 
						|
 | 
						|
buildDesktopMenuFile :: DesktopEntry -> String
 | 
						|
buildDesktopMenuFile d = unlines ("[Desktop Entry]" : map keyvalue d) ++ "\n"
 | 
						|
  where
 | 
						|
	keyvalue (k, v) = k ++ "=" ++ toString v
 | 
						|
 | 
						|
writeDesktopMenuFile :: DesktopEntry -> String -> IO ()
 | 
						|
writeDesktopMenuFile d file = do
 | 
						|
	createDirectoryIfMissing True (parentDir file)
 | 
						|
	writeFile file $ buildDesktopMenuFile d
 | 
						|
 | 
						|
{- Path to use for a desktop menu file, in either the systemDataDir or
 | 
						|
 - the userDataDir -}
 | 
						|
desktopMenuFilePath :: String -> FilePath -> FilePath
 | 
						|
desktopMenuFilePath basename datadir = 
 | 
						|
	datadir </> "applications" </> desktopfile basename
 | 
						|
 | 
						|
{- Path to use for a desktop autostart file, in either the systemDataDir
 | 
						|
 - or the userDataDir -}
 | 
						|
autoStartPath :: String -> FilePath -> FilePath
 | 
						|
autoStartPath basename configdir =
 | 
						|
	configdir </> "autostart" </> desktopfile basename
 | 
						|
 | 
						|
desktopfile :: FilePath -> FilePath
 | 
						|
desktopfile f = f ++ ".desktop"
 | 
						|
 | 
						|
{- Directory used for installation of system wide data files.. -}
 | 
						|
systemDataDir :: FilePath
 | 
						|
systemDataDir = "/usr/share"
 | 
						|
 | 
						|
{- Directory used for installation of system wide config files. -}
 | 
						|
systemConfigDir :: FilePath
 | 
						|
systemConfigDir = "/etc/xdg"
 | 
						|
 | 
						|
{- Directory for user data files. -}
 | 
						|
userDataDir :: IO FilePath
 | 
						|
userDataDir = xdgEnvHome "DATA_HOME" ".local/share"
 | 
						|
 | 
						|
{- Directory for user config files. -}
 | 
						|
userConfigDir :: IO FilePath
 | 
						|
userConfigDir = xdgEnvHome "CONFIG_HOME" ".config"
 | 
						|
 | 
						|
{- Directory for the user's Desktop, may be localized. 
 | 
						|
 -
 | 
						|
 - This is not looked up very fast; the config file is in a shell format
 | 
						|
 - that is best parsed by shell, so xdg-user-dir is used, with a fallback
 | 
						|
 - to ~/Desktop. -}
 | 
						|
userDesktopDir :: IO FilePath
 | 
						|
userDesktopDir = maybe fallback return =<< (parse <$> xdg_user_dir)
 | 
						|
  where
 | 
						|
	parse = maybe Nothing (headMaybe . lines)
 | 
						|
	xdg_user_dir = catchMaybeIO $ readProcess "xdg-user-dir" ["DESKTOP"]
 | 
						|
	fallback = xdgEnvHome "DESKTOP_DIR" "Desktop"
 | 
						|
 | 
						|
xdgEnvHome :: String -> String -> IO String
 | 
						|
xdgEnvHome envbase homedef = do
 | 
						|
	home <- myHomeDir
 | 
						|
	catchDefaultIO (home </> homedef) $
 | 
						|
		getEnv $ "XDG_" ++ envbase
 |