diff options
Diffstat (limited to 'accounts/gkleen@sif/xmonad/xmonad.hs')
| -rw-r--r-- | accounts/gkleen@sif/xmonad/xmonad.hs | 898 |
1 files changed, 898 insertions, 0 deletions
diff --git a/accounts/gkleen@sif/xmonad/xmonad.hs b/accounts/gkleen@sif/xmonad/xmonad.hs new file mode 100644 index 00000000..f3a59f34 --- /dev/null +++ b/accounts/gkleen@sif/xmonad/xmonad.hs | |||
| @@ -0,0 +1,898 @@ | |||
| 1 | {-# LANGUAGE TupleSections, ViewPatterns, OverloadedStrings, FlexibleInstances, UndecidableInstances, MultiWayIf #-} | ||
| 2 | |||
| 3 | import XMonad | ||
| 4 | import XMonad.Hooks.DynamicLog | ||
| 5 | import XMonad.Hooks.ManageDocks | ||
| 6 | import XMonad.Util.Run | ||
| 7 | import XMonad.Util.Loggers | ||
| 8 | import XMonad.Util.EZConfig(additionalKeys) | ||
| 9 | import System.IO | ||
| 10 | import System.IO.Error | ||
| 11 | import System.Environment | ||
| 12 | import Data.Map (Map) | ||
| 13 | import qualified Data.Map as Map | ||
| 14 | import qualified XMonad.StackSet as W | ||
| 15 | import System.Exit | ||
| 16 | import Control.Monad.State (get) | ||
| 17 | -- import XMonad.Layout.Spiral | ||
| 18 | import Data.Ratio | ||
| 19 | import Data.List | ||
| 20 | import Data.Char | ||
| 21 | import Data.Maybe (fromMaybe, listToMaybe, maybeToList, catMaybes, isJust) | ||
| 22 | import XMonad.Layout.Tabbed | ||
| 23 | import XMonad.Prompt | ||
| 24 | import XMonad.Prompt.Input | ||
| 25 | import XMonad.Util.Scratchpad | ||
| 26 | import XMonad.Util.NamedScratchpad | ||
| 27 | import Control.Monad (sequence, liftM, liftM2, join, void) | ||
| 28 | import XMonad.Util.WorkspaceCompare | ||
| 29 | import XMonad.Layout.NoBorders | ||
| 30 | import XMonad.Layout.PerWorkspace | ||
| 31 | import XMonad.Layout.SimplestFloat | ||
| 32 | import XMonad.Layout.Renamed | ||
| 33 | import XMonad.Layout.Reflect | ||
| 34 | import XMonad.Layout.OnHost | ||
| 35 | import XMonad.Layout.Combo | ||
| 36 | import XMonad.Layout.ComboP | ||
| 37 | import XMonad.Layout.Column | ||
| 38 | import XMonad.Layout.TwoPane | ||
| 39 | import XMonad.Layout.IfMax | ||
| 40 | import XMonad.Layout.LayoutBuilder | ||
| 41 | import XMonad.Layout.WindowNavigation | ||
| 42 | import XMonad.Layout.Dwindle | ||
| 43 | import XMonad.Layout.TrackFloating | ||
| 44 | import System.Process | ||
| 45 | import System.Directory (removeFile) | ||
| 46 | import System.Posix.Files | ||
| 47 | import System.FilePath ((</>)) | ||
| 48 | import Control.Concurrent | ||
| 49 | import System.Posix.Process (getProcessID) | ||
| 50 | import System.IO.Error | ||
| 51 | import System.IO | ||
| 52 | import XMonad.Hooks.ManageHelpers hiding (CW) | ||
| 53 | import XMonad.Hooks.UrgencyHook as U | ||
| 54 | import XMonad.Hooks.EwmhDesktops | ||
| 55 | import XMonad.StackSet (RationalRect (..)) | ||
| 56 | import Control.Monad (when, filterM, (<=<)) | ||
| 57 | import Graphics.X11.ExtraTypes.XF86 | ||
| 58 | import XMonad.Util.Cursor | ||
| 59 | import XMonad.Actions.Warp | ||
| 60 | import XMonad.Actions.FloatKeys | ||
| 61 | import XMonad.Util.SpawnOnce | ||
| 62 | import System.Directory | ||
| 63 | import System.FilePath | ||
| 64 | import XMonad.Actions.CopyWindow | ||
| 65 | import XMonad.Hooks.ServerMode | ||
| 66 | import XMonad.Actions.Commands | ||
| 67 | import XMonad.Actions.CycleWS | ||
| 68 | import XMonad.Actions.RotSlaves | ||
| 69 | import XMonad.Actions.UpdatePointer | ||
| 70 | import XMonad.Prompt.Window | ||
| 71 | import Data.IORef | ||
| 72 | import Data.Monoid | ||
| 73 | import Data.String | ||
| 74 | import qualified XMonad.Actions.PhysicalScreens as P | ||
| 75 | |||
| 76 | import XMonad.Layout.IM | ||
| 77 | |||
| 78 | import XMonad.Prompt.MyShell | ||
| 79 | import XMonad.Prompt.MyPass | ||
| 80 | import XMonad.Prompt.MySsh | ||
| 81 | |||
| 82 | import XMonad.Mpv | ||
| 83 | |||
| 84 | import Network.HostName | ||
| 85 | |||
| 86 | import Control.Applicative ((<$>)) | ||
| 87 | |||
| 88 | import Libnotify as Notify hiding (appName) | ||
| 89 | import qualified Libnotify as Notify (appName) | ||
| 90 | import Libnotify (Notification) | ||
| 91 | -- import System.Information.Battery | ||
| 92 | |||
| 93 | import Data.Int (Int32) | ||
| 94 | |||
| 95 | import System.Posix.Process | ||
| 96 | import System.Posix.Signals | ||
| 97 | import System.Posix.IO as Posix | ||
| 98 | import Control.Exception | ||
| 99 | |||
| 100 | import System.IO.Unsafe | ||
| 101 | |||
| 102 | import Control.Monad.Trans.Class | ||
| 103 | import Control.Monad.Trans.Maybe | ||
| 104 | |||
| 105 | import Data.Fixed (Micro) | ||
| 106 | |||
| 107 | import qualified Data.Text as Text | ||
| 108 | import Data.Ord (comparing) | ||
| 109 | import Debug.Trace | ||
| 110 | |||
| 111 | instance MonadIO m => IsString (m ()) where | ||
| 112 | fromString = spawn | ||
| 113 | |||
| 114 | type KeyMap = Map (ButtonMask, KeySym) (X ()) | ||
| 115 | |||
| 116 | data Host = Host | ||
| 117 | { hName :: HostName | ||
| 118 | , hManageHook :: ManageHook | ||
| 119 | , hWsp :: Integer -> WorkspaceId | ||
| 120 | , hCoWsp :: String -> Maybe WorkspaceId | ||
| 121 | , hKeysMod :: XConfig Layout -> (KeyMap -> KeyMap) | ||
| 122 | , hScreens :: [P.PhysicalScreen] | ||
| 123 | , hKbLayouts :: [(String, Maybe String)] | ||
| 124 | , hCmds :: X [(String, X ())] | ||
| 125 | , hKeyUpKeys :: XConfig Layout -> KeyMap | ||
| 126 | } | ||
| 127 | |||
| 128 | defaultHost = Host { hName = "unkown" | ||
| 129 | , hManageHook = composeOne [manageScratchTerm] | ||
| 130 | , hWsp = show | ||
| 131 | , hCoWsp = const Nothing | ||
| 132 | , hKeysMod = const id | ||
| 133 | , hScreens = [0,1..] | ||
| 134 | , hKbLayouts = [ ("us", Just "dvp") | ||
| 135 | , ("us", Nothing) | ||
| 136 | , ("de", Nothing) | ||
| 137 | ] | ||
| 138 | , hCmds = return [] | ||
| 139 | , hKeyUpKeys = const Map.empty | ||
| 140 | } | ||
| 141 | |||
| 142 | browser :: String | ||
| 143 | browser = "env MOZ_USE_XINPUT2=1 firefox" | ||
| 144 | |||
| 145 | hostFromName :: HostName -> Host | ||
| 146 | hostFromName h@("vali") = defaultHost { hName = h | ||
| 147 | , hManageHook = composeOne $ catMaybes [ Just manageScratchTerm | ||
| 148 | , assign "web" $ className =? ".dwb-wrapped" | ||
| 149 | , assign "web" $ className =? "Chromium" | ||
| 150 | , assign "work" $ className =? "Emacs" | ||
| 151 | , assign "media" $ className =? "mpv" | ||
| 152 | ] | ||
| 153 | , hWsp = hWsp | ||
| 154 | , hCoWsp = hCoWsp | ||
| 155 | , hKeysMod = \conf -> Map.union $ (Map.fromList $ join $ map (spawnBindings conf) [ (xK_d, ["chromium", "chromium $(xclip -o)"]) | ||
| 156 | , (xK_e, ["emacsclient -c"]) | ||
| 157 | ]) | ||
| 158 | `Map.union` | ||
| 159 | ( Map.fromList [ ((XMonad.modMask conf .|. controlMask, xK_Return), scratchpadSpawnActionCustom $ (XMonad.terminal conf) ++ " -name scratchpad -title scratchpad -e tmux new-session -D -s scratch") | ||
| 160 | ] ) | ||
| 161 | , hScreens = hScreens defaultHost | ||
| 162 | } | ||
| 163 | where | ||
| 164 | workspaceNames = Map.fromList [ (2, "web") | ||
| 165 | , (3, "work") | ||
| 166 | , (10, "media") | ||
| 167 | ] | ||
| 168 | hWsp = wspFromMap workspaceNames | ||
| 169 | hCoWsp = coWspFromMap workspaceNames | ||
| 170 | assign wsp test = (\wsp -> test -?> doShift wsp) <$> hCoWsp wsp | ||
| 171 | hostFromName h | ||
| 172 | | h `elem` ["hel", "sif"] = defaultHost { hName = h | ||
| 173 | , hManageHook = namedScratchpadManageHook scratchpads <+> composeOne (catMaybes | ||
| 174 | [ assign "mpv" $ className =? "mpv" | ||
| 175 | , assign "mpv" $ (className =? "URxvt" <&&> title =? "irssi") | ||
| 176 | , assign "mpv" $ (className =? "URxvt" <&&> resource =? "presentation") | ||
| 177 | , assign "mpv" $ stringProperty "WM_WINDOW_ROLE" =? "presentation" | ||
| 178 | , assign "read" $ stringProperty "WM_WINDOW_ROLE" =? "presenter" | ||
| 179 | , assign "mpv" $ className =? "factorio" | ||
| 180 | , assign "web" $ className =? "chromium-browser" | ||
| 181 | , assign "web" $ className =? "Google-chrome" | ||
| 182 | , assign "work" $ (appName =? "Devtools" <&&> className =? "Firefox") | ||
| 183 | , assign "work" $ className =? "Postman" | ||
| 184 | , assign "web" $ className =? "Firefox" | ||
| 185 | , assign "comm" $ (className =? "URxvt" <&&> resource =? "comm") | ||
| 186 | , assign "comm" $ (className =? "Emacs" <&&> title =? "Mail") | ||
| 187 | , assign "comm" $ className =? "Zulip" | ||
| 188 | , assign "comm" $ className =? "Discord" | ||
| 189 | , assign "media" $ (className =? "URxvt" <&&> resource =? "media") | ||
| 190 | , assign "media" $ (className =? "URxvt" <&&> title =? "streamlink") | ||
| 191 | , assign "media" $ (className =? "URxvt" <&&> title =? "mpv") | ||
| 192 | , assign "monitor" $ (className =? "URxvt" <&&> fmap ("monitor" `isInfixOf`) title) | ||
| 193 | , assign "monitor" $ className =? "Grafana" | ||
| 194 | , Just $ (className =? "URxvt" <&&> resource =? "htop") -?> centerFloat | ||
| 195 | , Just $ (className =? "Scp-dbus-service.py") -?> centerFloat | ||
| 196 | , Just $ (className =? "URxvt" <&&> resource =? "log") -?> centerFloat | ||
| 197 | , assign "work" $ className =? "URxvt" | ||
| 198 | , assign' ["work", "uni"] $ (className =? "Emacs" <&&> appName /=? "Edit_with_Emacs_FRAME") | ||
| 199 | , assign' ["work", "uni"] $ className =? "jetbrains-idea-ce" | ||
| 200 | , assign "read" $ className =? "llpp" | ||
| 201 | , assign "read" $ className =? "Evince" | ||
| 202 | , assign "read" $ fmap ("zathura" `isInfixOf`) title | ||
| 203 | , assign "read" $ className =? "MuPDF" | ||
| 204 | , assign "read" $ className =? "Xournal" | ||
| 205 | , assign "read" $ appName =? "com-trollworks-gcs-app-GCS" | ||
| 206 | , assign "read" $ appName =? "Tux.py" | ||
| 207 | , assign "read" $ className =? "Gnucash" | ||
| 208 | , assign "comm" $ className =? "Skype" | ||
| 209 | , assign "comm" $ className =? "Daily" | ||
| 210 | , assign "comm" $ className =? "Pidgin" | ||
| 211 | , assign "comm" $ className =? "Slack" | ||
| 212 | , Just $ (resource =? "xvkbd") -?> doRectFloat $ RationalRect (1 % 8) (3 % 8) (6 % 8) (4 % 8) | ||
| 213 | , Just $ (stringProperty "_NET_WM_WINDOW_TYPE" =? "_NET_WM_WINDOW_TYPE_DIALOG") -?> doFloat | ||
| 214 | , Just $ (className =? "Dunst") -?> doFloat | ||
| 215 | , Just $ (className =? "Xmessage") -?> doCenterFloat | ||
| 216 | , Just $ (className =? "Nm-openconnect-auth-dialog") -?> centerFloat | ||
| 217 | , Just $ (className =? "Pinentry") -?> doCenterFloat | ||
| 218 | , Just $ (className =? "pinentry") -?> doCenterFloat | ||
| 219 | , Just $ (appName =? "Edit_with_Emacs_FRAME") -?> centerFloat | ||
| 220 | , Just $ (stringProperty "WM_WINDOW_ROLE" =? "GtkFileChooseDialog") -?> centerFloatSmall | ||
| 221 | , Just $ (className =? "Nvidia-settings") -?> doCenterFloat | ||
| 222 | , Just $ fmap ("Minetest" `isInfixOf`) title -?> doIgnore | ||
| 223 | , Just $ fmap ("Automachef" `isInfixOf`) title -?> doIgnore | ||
| 224 | ]) | ||
| 225 | , hWsp = hWsp | ||
| 226 | , hCoWsp = hCoWsp | ||
| 227 | , hKeysMod = \conf -> Map.union $ (Map.fromList $ join $ map (spawnBindings conf) [ (xK_e, ["emacsclient -c"]) | ||
| 228 | , (xK_d, [fromString browser, fromString $ browser ++ " $(xclip -o)", fromString $ "notmuch-links"]) | ||
| 229 | , (xK_f, ["urxvtc -name comm -title Feeds -e mosh odin -- tmux new-session -ADs comm"]) | ||
| 230 | , (xK_c, [ inputPrompt xPConfig "dc" ?+ dc ]) | ||
| 231 | , (xK_g, ["pidgin"]) | ||
| 232 | , (xK_s, ["skype"]) | ||
| 233 | -- , (xK_p, [mkPassPrompt "Type password" pwType xPConfig, mkPassPrompt "Show password" pwShow xPConfig, mkPassPrompt "Copy password" pwClip xPConfig]) | ||
| 234 | , (xK_w, ["sudo rewacom"]) | ||
| 235 | , (xK_y, [ "tmux new-window -dt media /var/media/link.hs $(xclip -o)" | ||
| 236 | , "urxvtc -name media -e tmuxp load /var/media" | ||
| 237 | ]) | ||
| 238 | , (xK_l, [ "tmux new-window -dt media mpv $(xclip -o)" | ||
| 239 | , "tmux new-window -dt media streamlink --retry-open 10 $(xclip -o)" | ||
| 240 | ]) | ||
| 241 | , (xK_m, [ "emacsclient -c -F \"'(title . \\\"Mail\\\")\" -e '(notmuch)'" | ||
| 242 | , "emacsclient -c -F \"'(title . \\\"Mail\\\")\" -e '(notmuch-mua-new-mail)'" | ||
| 243 | , "emacsclient -c -F \"'(title . \\\"Mail\\\")\" -e \"(browse-url-mail \"$(xclip -o)\")\"" | ||
| 244 | ]) | ||
| 245 | , (xK_Return, ["keynav start,windowzoom", "keynav start"]) | ||
| 246 | , (xK_t, [inputPrompt xPConfig "fuzzytime timer" ?+ fuzzytime, fuzzytime "unset", work_fuzzytime]) | ||
| 247 | , (xK_a, [inputPrompt xPConfig "adjmix" ?+ adjmix]) | ||
| 248 | , (xK_s, [ inputPromptWithCompl xPConfig "start synergy" synergyCompl ?+ synergyStart | ||
| 249 | , inputPromptWithCompl xPConfig "stop synergy" synergyCompl ?+ synergyStop | ||
| 250 | ]) | ||
| 251 | , (xK_h, [ "urxvtc -name htop -e htop" | ||
| 252 | , "urxvtc -name log -e journalctl -xef" | ||
| 253 | ]) | ||
| 254 | , (xK_x, [ "autorandr -c" | ||
| 255 | , "autorandr -fl default" | ||
| 256 | ]) | ||
| 257 | , (xK_z, [ "zulip -- --force-device-scale-factor=2" | ||
| 258 | ]) | ||
| 259 | ]) | ||
| 260 | `Map.union` | ||
| 261 | ( Map.fromList [ ((XMonad.modMask conf .|. controlMask, xK_Return), namedScratchpadAction scratchpads "term") | ||
| 262 | , ((XMonad.modMask conf .|. controlMask, xK_a), namedScratchpadAction scratchpads "pavucontrol") | ||
| 263 | , ((XMonad.modMask conf .|. controlMask, xK_w), namedScratchpadAction scratchpads "alarms") | ||
| 264 | , ((XMonad.modMask conf .|. controlMask, xK_b), namedScratchpadAction scratchpads "blueman") | ||
| 265 | , ((XMonad.modMask conf .|. controlMask, xK_p), namedScratchpadAction scratchpads "keepassxc") | ||
| 266 | , ((XMonad.modMask conf .|. controlMask, xK_t), namedScratchpadAction scratchpads "toggl") | ||
| 267 | , ((XMonad.modMask conf .|. controlMask, xK_e), namedScratchpadAction scratchpads "emacs") | ||
| 268 | , ((XMonad.modMask conf .|. controlMask, xK_m), namedScratchpadAction scratchpads "calendar") | ||
| 269 | , ((XMonad.modMask conf .|. controlMask, xK_f), namedScratchpadAction scratchpads "music") | ||
| 270 | , ((XMonad.modMask conf .|. mod1Mask, xK_Up), rotate U) | ||
| 271 | , ((XMonad.modMask conf .|. mod1Mask, xK_Down), rotate D) | ||
| 272 | , ((XMonad.modMask conf .|. mod1Mask, xK_Left), rotate L) | ||
| 273 | , ((XMonad.modMask conf .|. mod1Mask, xK_Right), rotate R) | ||
| 274 | -- , ((XMonad.modMask conf .|. shiftMask, xK_a), startMute "hel") | ||
| 275 | ] ) | ||
| 276 | , hKeyUpKeys = \conf -> Map.fromList [ -- ((XMonad.modMask conf .|. shiftMask, xK_a), stopMute "hel") | ||
| 277 | ] | ||
| 278 | , hScreens = hScreens defaultHost | ||
| 279 | , hCmds = return [ ("prev-workspace", prevWS) | ||
| 280 | , ("next-workspace", nextWS) | ||
| 281 | , ("prev-window", rotAllDown) | ||
| 282 | , ("next-window", rotAllUp) | ||
| 283 | , ("banish", banishScreen LowerRight) | ||
| 284 | , ("update-gpg-tty", safeSpawn "gpg-connect-agent" ["UPDATESTARTUPTTY", "/bye"]) | ||
| 285 | , ("rescreen", rescreen) | ||
| 286 | , ("repanel", do | ||
| 287 | spawn "nm-applet" | ||
| 288 | spawn "blueman-applet" | ||
| 289 | spawn "pasystray" | ||
| 290 | spawn "kdeconnect-indicator" | ||
| 291 | spawn "dunst -print" | ||
| 292 | spawn "udiskie" | ||
| 293 | spawn "autocutsel -s PRIMARY" | ||
| 294 | spawn "autocutsel -s CLIPBOARD" | ||
| 295 | ) | ||
| 296 | , ("pause", mediaMpv $ MpvSetProperty "pause" True) | ||
| 297 | , ("unpause", mediaMpv $ MpvSetProperty "pause" False) | ||
| 298 | , ("exit", io $ exitWith ExitSuccess) | ||
| 299 | ] | ||
| 300 | } | ||
| 301 | where | ||
| 302 | withGdkScale act = void . xfork $ setEnv "GDK_SCALE" "2" >> act | ||
| 303 | workspaceNames = Map.fromList [ (1, "comm") | ||
| 304 | , (2, "web") | ||
| 305 | , (3, "work") | ||
| 306 | , (4, "read") | ||
| 307 | , (5, "monitor") | ||
| 308 | , (6, "uni") | ||
| 309 | , (9, "media") | ||
| 310 | , (10, "mpv") | ||
| 311 | ] | ||
| 312 | scratchpads = [ NS "term" "urxvtc -name scratchpad -title scratchpad -e tmux new-session -AD -s scratch" (resource =? "scratchpad") centerFloat | ||
| 313 | , NS "pavucontrol" "pavucontrol" (resource =? "pavucontrol") centerFloat | ||
| 314 | , NS "alarms" "alarm-clock-applet" (className =? "Alarm-clock-applet" <&&> title =? "Alarms") centerFloat | ||
| 315 | , NS "blueman" "blueman-manager" (className =? ".blueman-manager-wrapped") centerFloat | ||
| 316 | , NS "keepassxc" "keepassxc" (className =? "KeePassXC") centerFloat | ||
| 317 | , NS "toggl" "toggldesktop" (className =? "Toggl Desktop") centerFloat | ||
| 318 | , NS "calendar" "minetime -- --force-device-scale-factor=1.6" (className =? "MineTime") centerFloat | ||
| 319 | , NS "emacs" "emacsclient -c -F \"'(title . \\\"Scratchpad\\\")\"" (className =? "Emacs" <&&> title =? "Scratchpad") centerFloat | ||
| 320 | , NS "music" "google-play-music-desktop-player --force-device-scale-factor=1.6" (className =? "Google Play Music Desktop Player") centerFloat | ||
| 321 | ] | ||
| 322 | centerFloat = customFloating $ RationalRect (1 % 16) (1 % 16) (7 % 8) (7 % 8) | ||
| 323 | centerFloatSmall = customFloating $ RationalRect (1 % 4) (1 % 4) (1 % 2) (1 % 2) | ||
| 324 | hWsp = wspFromMap workspaceNames | ||
| 325 | hCoWsp = coWspFromMap workspaceNames | ||
| 326 | assign wsp test = (\wsp -> test -?> doShift wsp) <$> hCoWsp wsp | ||
| 327 | assign' :: [String] -> Query Bool -> Maybe MaybeManageHook | ||
| 328 | assign' wsps test = do | ||
| 329 | wsIds <- mapM hCoWsp wsps | ||
| 330 | return $ test -?> go wsIds | ||
| 331 | where | ||
| 332 | go :: [WorkspaceId] -> ManageHook | ||
| 333 | go wsps = do | ||
| 334 | visWsps <- liftX $ (\wset -> W.tag . W.workspace <$> W.current wset : W.visible wset) <$> gets windowset | ||
| 335 | case (filter (`elem` visWsps) wsps, wsps) of | ||
| 336 | (wsp : _, _) -> doShift wsp | ||
| 337 | (_, wsp : _) -> doShift wsp | ||
| 338 | ([], []) -> return mempty | ||
| 339 | rotate rot = do | ||
| 340 | safeSpawn "xrandr" ["--output", "eDP-1", "--rotate", xrandrDir] | ||
| 341 | mapM_ rotTouch touchscreens | ||
| 342 | where | ||
| 343 | xrandrDir = case rot of | ||
| 344 | U -> "normal" | ||
| 345 | L -> "left" | ||
| 346 | R -> "right" | ||
| 347 | D -> "inverted" | ||
| 348 | matrix = case rot of | ||
| 349 | U -> [ [ 1, 0, 0] | ||
| 350 | , [ 0, 1, 0] | ||
| 351 | , [ 0, 0, 1] | ||
| 352 | ] | ||
| 353 | L -> [ [ 0, -1, 1] | ||
| 354 | , [ 1, 0, 0] | ||
| 355 | , [ 0, 0, 1] | ||
| 356 | ] | ||
| 357 | R -> [ [ 0, 1, 0] | ||
| 358 | , [-1, 0, 1] | ||
| 359 | , [ 0, 0, 1] | ||
| 360 | ] | ||
| 361 | D -> [ [-1, 0, 1] | ||
| 362 | , [ 0, -1, 1] | ||
| 363 | , [ 0, 0, 1] | ||
| 364 | ] | ||
| 365 | touchscreens = [ "Wacom Co.,Ltd. Pen and multitouch sensor Finger touch" | ||
| 366 | , "Wacom Co.,Ltd. Pen and multitouch sensor Pen stylus" | ||
| 367 | , "Wacom Co.,Ltd. Pen and multitouch sensor Pen eraser" | ||
| 368 | ] | ||
| 369 | rotTouch screen = do | ||
| 370 | safeSpawn "xinput" $ ["set-prop", screen, "Coordinate Transformation Matrix"] ++ map (\n -> show n ++ ",") (concat matrix) | ||
| 371 | safeSpawn "xinput" ["map-to-output", screen, "eDP-1"] | ||
| 372 | withPw f label = io . void . forkProcess $ do | ||
| 373 | uninstallSignalHandlers | ||
| 374 | void $ createSession | ||
| 375 | (dropWhileEnd isSpace -> pw) <- readCreateProcess (proc "pass" ["show", label]) "" | ||
| 376 | void $ f pw | ||
| 377 | pwType :: String -> X () | ||
| 378 | pwType = withPw $ readCreateProcess (proc "xdotool" ["type", "--clearmodifiers", "--file", "-"]) | ||
| 379 | pwClip label = safeSpawn "pass" ["show", "--clip", label] | ||
| 380 | pwShow :: String -> X () | ||
| 381 | pwShow = withPw $ \pw -> do | ||
| 382 | xmessage <- fromMaybe "xmessage" <$> liftIO (lookupEnv "XMONAD_XMESSAGE") | ||
| 383 | readCreateProcess (proc xmessage ["-file", "-"]) pw | ||
| 384 | fuzzytime str = safeSpawn "fuzzytime" $ "timer" : words str | ||
| 385 | work_fuzzytime = io . void . forkProcess $ do | ||
| 386 | readCreateProcess (proc "worktime" []) "" >>= safeSpawn "fuzzytime" . ("timer" : ) . pure | ||
| 387 | adjmix str = safeSpawn "adjmix" $ words str | ||
| 388 | dc expr = void . xfork $ do | ||
| 389 | result <- readProcess "dc" [] $ expr ++ "f" | ||
| 390 | let | ||
| 391 | (first : rest) = filter (not . null) $ lines result | ||
| 392 | notification = Notify.summary first <> Notify.body (unlines rest) <> Notify.timeout Infinite <> Notify.urgency Normal <> Notify.appName "dc" | ||
| 393 | void $ Notify.display notification | ||
| 394 | synergyCompl = mkComplFunFromList' ["mathw86"] | ||
| 395 | synergyStart host = safeSpawn "systemctl" ["--user", "start", "synergy-rtunnel@" ++ host ++ ".service"] | ||
| 396 | synergyStop host = safeSpawn "systemctl" ["--user", "stop", "synergy-rtunnel@" ++ host ++ ".service"] | ||
| 397 | |||
| 398 | hostFromName _ = defaultHost | ||
| 399 | |||
| 400 | -- muteRef :: IORef (Maybe (String, Notification)) | ||
| 401 | -- {-# NOINLINE muteRef #-} | ||
| 402 | -- muteRef = unsafePerformIO $ newIORef Nothing | ||
| 403 | |||
| 404 | -- startMute, stopMute :: String -> X () | ||
| 405 | -- startMute sink = liftIO $ do | ||
| 406 | -- muted <- isJust <$> readIORef muteRef | ||
| 407 | -- when (not muted) $ do | ||
| 408 | -- let | ||
| 409 | -- notification = Notify.summary "Muted" <> Notify.timeout Infinite <> Notify.urgency Normal | ||
| 410 | -- level = "0.0dB" | ||
| 411 | -- -- level <- runProcessWithInput "ssh" ["bragi", "cat", "/dev/shm/mix/" ++ sink ++ "/level"] "" | ||
| 412 | -- -- callProcess "ssh" ["bragi", "adjmix", "-t", sink, "-o", "0"] | ||
| 413 | -- hPutStrLn stderr "Mute" | ||
| 414 | -- writeIORef muteRef . Just . (level, ) =<< Notify.display notification | ||
| 415 | -- stopMute sink = liftIO $ do | ||
| 416 | -- let | ||
| 417 | -- unmute (Just (level, notification)) = do | ||
| 418 | -- hPutStrLn stderr "Unmute" | ||
| 419 | -- -- callProcess "ssh" ["bragi", "adjmix", "-t", sink, "-o", level] | ||
| 420 | -- Notify.close notification | ||
| 421 | -- unmute Nothing = return () | ||
| 422 | -- muted <- isJust <$> readIORef muteRef | ||
| 423 | -- when muted . join . atomicModifyIORef muteRef $ (Nothing, ) . unmute | ||
| 424 | |||
| 425 | wspFromMap workspaceNames = \i -> case Map.lookup i workspaceNames of | ||
| 426 | Just str -> show i ++ " " ++ str | ||
| 427 | Nothing -> show i | ||
| 428 | |||
| 429 | coWspFromMap workspaceNames = \str -> case filter ((== str) . snd) $ Map.toList workspaceNames of | ||
| 430 | [] -> Nothing | ||
| 431 | [(i, _)] -> Just $ wspFromMap workspaceNames i | ||
| 432 | _ -> Nothing | ||
| 433 | |||
| 434 | spawnModifiers = [0, controlMask, shiftMask .|. controlMask] | ||
| 435 | spawnBindings :: XConfig layout -> (KeySym, [X ()]) -> [((KeyMask, KeySym), X ())] | ||
| 436 | spawnBindings conf (k, cmds) = zipWith (\m cmd -> ((modm .|. mod1Mask .|. m, k), cmd)) spawnModifiers cmds | ||
| 437 | where | ||
| 438 | modm = XMonad.modMask conf | ||
| 439 | |||
| 440 | manageScratchTerm = (resource =? "scratchpad" <||> resource =? "keysetup") -?> doRectFloat $ RationalRect (1 % 16) (1 % 16) (7 % 8) (7 % 8) | ||
| 441 | |||
| 442 | tabbedLayout t = renamed [Replace "Tabbed"] $ reflectHoriz $ t CustomShrink $ tabbedTheme | ||
| 443 | tabbedLayoutHoriz t = renamed [Replace "Tabbed Horiz"] $ reflectVert $ t CustomShrink $ tabbedTheme | ||
| 444 | tabbedTheme = def | ||
| 445 | { activeColor = "black" | ||
| 446 | , inactiveColor = "black" | ||
| 447 | , urgentColor = "black" | ||
| 448 | , activeBorderColor = "grey" | ||
| 449 | , inactiveBorderColor = "#202020" | ||
| 450 | , urgentBorderColor = "#bb0000" | ||
| 451 | , activeTextColor = "grey" | ||
| 452 | , inactiveTextColor = "grey" | ||
| 453 | , urgentTextColor = "grey" | ||
| 454 | , decoHeight = 32 | ||
| 455 | , fontName = "xft:Fira Mono for Powerline:style=Medium:pixelsize=22.5" | ||
| 456 | } | ||
| 457 | |||
| 458 | main :: IO () | ||
| 459 | main = do | ||
| 460 | arguments <- either (const []) id <$> tryIOError getArgs | ||
| 461 | case arguments of | ||
| 462 | ["--command", s] -> do | ||
| 463 | d <- openDisplay "" | ||
| 464 | rw <- rootWindow d $ defaultScreen d | ||
| 465 | a <- internAtom d "XMONAD_COMMAND" False | ||
| 466 | m <- internAtom d s False | ||
| 467 | allocaXEvent $ \e -> do | ||
| 468 | setEventType e clientMessage | ||
| 469 | setClientMessageEvent e rw a 32 m currentTime | ||
| 470 | sendEvent d rw False structureNotifyMask e | ||
| 471 | sync d False | ||
| 472 | _ -> do | ||
| 473 | -- batteryMon <- xfork $ monitorBattery Nothing Nothing | ||
| 474 | hostname <- getHostName | ||
| 475 | let | ||
| 476 | host = hostFromName hostname | ||
| 477 | setEnv "HOST" hostname | ||
| 478 | let myConfig = withHostUrgency . ewmh $ docks def | ||
| 479 | { manageHook = hManageHook host | ||
| 480 | , terminal = "urxvtc" | ||
| 481 | , layoutHook = smartBorders . avoidStruts $ windowNavigation layout' | ||
| 482 | , logHook = do | ||
| 483 | dynamicLogString xmobarPP' >>= writeProps | ||
| 484 | updatePointer (99 % 100, 98 % 100) (0, 0) | ||
| 485 | , modMask = mod4Mask | ||
| 486 | , keys = \conf -> hKeysMod host conf $ myKeys' conf host | ||
| 487 | , workspaces = take (length numKeys) $ map wsp [1..] | ||
| 488 | , startupHook = setDefaultCursor xC_left_ptr | ||
| 489 | , normalBorderColor = "#202020" | ||
| 490 | , focusedBorderColor = "grey" | ||
| 491 | , handleEventHook = fullscreenEventHook <+> (serverModeEventHookCmd' $ hCmds host) <+> keyUpEventHook | ||
| 492 | } | ||
| 493 | writeProps str = do | ||
| 494 | let encodeCChar = map $ fromIntegral . fromEnum | ||
| 495 | atoms = [ "_XMONAD_WORKSPACES" | ||
| 496 | , "_XMONAD_LAYOUT" | ||
| 497 | , "_XMONAD_TITLE" | ||
| 498 | ] | ||
| 499 | (flip mapM_) (zip atoms (lines str)) $ \(atom', content) -> do | ||
| 500 | ustring <- getAtom "UTF8_STRING" | ||
| 501 | atom <- getAtom atom' | ||
| 502 | withDisplay $ \dpy -> io $ do | ||
| 503 | root <- rootWindow dpy $ defaultScreen dpy | ||
| 504 | changeProperty8 dpy root atom ustring propModeReplace $ encodeCChar content | ||
| 505 | sync dpy True | ||
| 506 | wsp = hWsp host | ||
| 507 | -- We can´t define per-host layout modifiers because we lack dependent types | ||
| 508 | layout' = onHost "skadhi" ( onWorkspace (wsp 1) (Full ||| withIM (1%5) (Title "Buddy List") tabbedLayout') $ | ||
| 509 | onWorkspace (wsp 10) Full $ | ||
| 510 | onWorkspace (wsp 2) (Full ||| tabbedLayout') $ | ||
| 511 | onWorkspace (wsp 5) tabbedLayout' $ | ||
| 512 | onWorkspace (wsp 8) (withIM (1%5) (Title "Friends") tabbedLayout') $ | ||
| 513 | defaultLayouts | ||
| 514 | ) $ | ||
| 515 | onHost "vali" ( onWorkspace (wsp 2) (Full ||| tabbedLayout' ||| combineTwo (TwoPane 0.01 0.57) Full tabbedLayout') $ | ||
| 516 | onWorkspace (wsp 3) workLayouts $ | ||
| 517 | defaultLayouts | ||
| 518 | ) $ | ||
| 519 | onHost "hel" ( onWorkspace (wsp 1) (withIM (1 % 8) (Title "Buddy List") $ trackFloating tabbedLayout') $ | ||
| 520 | onWorkspace (wsp 2) (tabbedLayout''' ||| Dwindle R CW 1 (5 % 100)) $ | ||
| 521 | onWorkspace (wsp 3) workLayouts $ | ||
| 522 | onWorkspace (wsp 6) workLayouts $ | ||
| 523 | onWorkspace (wsp 4) (tabbedLayout' ||| tabbedLayoutHoriz' ||| Dwindle R CW 1 (5 % 100)) $ | ||
| 524 | onWorkspace (wsp 5) (tabbedLayout''' ||| Dwindle R CW 1 (5 % 100)) $ | ||
| 525 | onWorkspace (wsp 10) (tabbedLayout''' ||| combineTwoP (TwoPane (1 % 100) (3 % 4)) tabbedLayout''' tabbedLayout''' (ClassName "mpv") ||| Dwindle R CW 1 (5 % 100)) $ | ||
| 526 | defaultLayouts | ||
| 527 | ) $ | ||
| 528 | onHost "sif" ( onWorkspace (wsp 1) (withIM (1 % 8) (Title "Buddy List") $ trackFloating tabbedLayout') $ | ||
| 529 | onWorkspace (wsp 2) (tabbedLayout''' ||| Dwindle R CW 1 (5 % 100)) $ | ||
| 530 | onWorkspace (wsp 3) workLayouts $ | ||
| 531 | onWorkspace (wsp 6) workLayouts $ | ||
| 532 | onWorkspace (wsp 4) (tabbedLayout' ||| tabbedLayoutHoriz' ||| Dwindle R CW 1 (5 % 100)) $ | ||
| 533 | onWorkspace (wsp 5) (tabbedLayout''' ||| Dwindle R CW 1 (5 % 100)) $ | ||
| 534 | onWorkspace (wsp 10) (tabbedLayout''' ||| combineTwoP (TwoPane (1 % 100) (3 % 4)) tabbedLayout''' tabbedLayout''' (ClassName "mpv") ||| Dwindle R CW 1 (5 % 100)) $ | ||
| 535 | defaultLayouts | ||
| 536 | ) $ | ||
| 537 | defaultLayouts | ||
| 538 | -- tabbedLayout''' = renamed [Replace "Tabbed'"] $ IfMax 1 (noBorders Full) (tabbedLayout tabbedBottomAlways) | ||
| 539 | tabbedLayout''' = tabbedLayout tabbedBottom | ||
| 540 | tabbedLayout' = tabbedLayout tabbedBottomAlways | ||
| 541 | tabbedLayoutHoriz' = tabbedLayoutHoriz tabbedLeftAlways | ||
| 542 | defaultLayouts = {- spiralWithDir East CW (1 % 2) -} Dwindle R CW 1 (5 % 100) ||| tabbedLayout' ||| Full | ||
| 543 | -- workLayouts = {- spiralWithDir East CW (1 % 2) -} Dwindle R CW (2 % 1) (5 % 100) ||| tabbedLayout' ||| Full | ||
| 544 | workLayouts = tabbedLayout' ||| (renamed [Replace "Combined"] $ combineTwoP (TwoPane (1 % 100) (1891 % 2560)) tabbedLayout''' (Column 1.6) (ClassName "Postman" `Or` ClassName "Emacs" `Or` ClassName "jetbrains-idea-ce" `Or` (Resource "Devtools" `And` ClassName "Firefox"))) ||| Full ||| Dwindle R CW 1 (5 % 100) | ||
| 545 | sqrtTwo = approxRational (sqrt 2) (1 / 2560) | ||
| 546 | xmobarPP' = xmobarPP { ppTitle = shorten 80 | ||
| 547 | , ppSort = (liftM2 (.)) getSortByIndex $ return scratchpadFilterOutWorkspace | ||
| 548 | , ppUrgent = wrap "(" ")" . xmobarColor "red" "" | ||
| 549 | , ppHiddenNoWindows = xmobarColor "#202020" "" . wrap "(" ")" | ||
| 550 | , ppVisible = wrap "(" ")" . xmobarColor "yellow" "" | ||
| 551 | , ppCurrent = wrap "(" ")" . xmobarColor "green" "" | ||
| 552 | , ppHidden = wrap "(" ")" | ||
| 553 | , ppWsSep = " " | ||
| 554 | , ppSep = "\n" | ||
| 555 | } | ||
| 556 | withHostUrgency = case hostname of | ||
| 557 | "hel" -> withUrgencyHookC urgencyHook' $ urgencyConfig { suppressWhen = U.Never, remindWhen = Dont } | ||
| 558 | "sif" -> withUrgencyHookC urgencyHook' $ urgencyConfig { suppressWhen = U.Never, remindWhen = Dont } | ||
| 559 | _ -> id | ||
| 560 | urgencyHook' window = do | ||
| 561 | runQuery ((resource =? "comm" <||> resource =? "Pidgin" <||> className =? "Gajim" <||> className =? "Skype") --> safeSpawn "thinklight" ["Blink", "100"]) window | ||
| 562 | urgencyHook (BorderUrgencyHook { urgencyBorderColor = "#bb0000" }) window | ||
| 563 | shutdown :: SomeException -> IO a | ||
| 564 | shutdown e = do | ||
| 565 | let pids = [ -- batteryMon | ||
| 566 | ] | ||
| 567 | mapM_ (signalProcess sigTERM) pids | ||
| 568 | mapM_ (getProcessStatus False False) pids | ||
| 569 | throw e | ||
| 570 | keyUpEventHook :: Event -> X All | ||
| 571 | keyUpEventHook event = handle event >> return (All True) | ||
| 572 | where | ||
| 573 | handle (KeyEvent { ev_event_type = t, ev_state = m, ev_keycode = code }) | ||
| 574 | | t == keyRelease = withDisplay $ \dpy -> do | ||
| 575 | s <- io $ keycodeToKeysym dpy code 0 | ||
| 576 | mClean <- cleanMask m | ||
| 577 | ks <- asks $ hKeyUpKeys host . config | ||
| 578 | userCodeDef () $ whenJust (Map.lookup (mClean, s) ks) id | ||
| 579 | | otherwise = return () | ||
| 580 | handle _ = return () | ||
| 581 | handle shutdown $ launch myConfig | ||
| 582 | |||
| 583 | secs :: Int -> Int | ||
| 584 | secs = (* 1000000) | ||
| 585 | |||
| 586 | -- monitorBattery :: Maybe BatteryContext -> Maybe Notification -> IO () | ||
| 587 | -- monitorBattery Nothing n = do | ||
| 588 | -- ctx <- batteryContextNew | ||
| 589 | -- case ctx of | ||
| 590 | -- Nothing -> threadDelay (secs 10) >> monitorBattery Nothing n | ||
| 591 | -- Just _ -> monitorBattery ctx n | ||
| 592 | -- monitorBattery ctx@(Just ctx') n = do | ||
| 593 | -- batInfo <- getBatteryInfo ctx' | ||
| 594 | -- case batInfo of | ||
| 595 | -- Nothing -> threadDelay (secs 1) >> monitorBattery ctx n | ||
| 596 | -- Just batInfo -> do | ||
| 597 | -- let n' | ||
| 598 | -- | batteryState batInfo == BatteryStateDischarging | ||
| 599 | -- , timeLeft <= 1200 | ||
| 600 | -- , timeLeft > 0 = Just $ summary "Discharging" <> hint "value" percentage <> urgency u <> body (duz timeLeft ++ "left") | ||
| 601 | -- | otherwise = Nothing | ||
| 602 | -- u | ||
| 603 | -- | timeLeft <= 600 = Critical | ||
| 604 | -- | timeLeft <= 1800 = Normal | ||
| 605 | -- | otherwise = Low | ||
| 606 | -- timeLeft = batteryTimeToEmpty batInfo | ||
| 607 | -- percentage :: Int32 | ||
| 608 | -- percentage = round $ batteryPercentage batInfo | ||
| 609 | -- ts = [("s", 60), ("m", 60), ("h", 24), ("d", 365), ("y", 1)] | ||
| 610 | -- duz ms = ss | ||
| 611 | -- where (ss, _) = foldl (\(ss, x) (s, y) -> ((if rem x y > 0 then show (rem x y) ++ s ++ " " else "") ++ ss , quot x y)) ("", ms) ts | ||
| 612 | -- case n' of | ||
| 613 | -- Just n' -> Notify.display (maybe mempty reuse n <> Notify.appName "monitorBattery" <> n') >>= (\n -> threadDelay (secs 2) >> monitorBattery ctx (Just n)) | ||
| 614 | -- Nothing -> threadDelay (secs 30) >> monitorBattery ctx n | ||
| 615 | |||
| 616 | disableTouchpad, disableTrackpoint, enableTrackpoint, enableTouchpad :: X () | ||
| 617 | enableTouchpad = safeSpawn "xinput" ["enable", "SynPS/2 Synaptics TouchPad"] | ||
| 618 | disableTouchpad = safeSpawn "xinput" ["disable", "SynPS/2 Synaptics TouchPad"] | ||
| 619 | enableTrackpoint = safeSpawn "xinput" ["enable", "TPPS/2 IBM TrackPoint"] | ||
| 620 | disableTrackpoint = safeSpawn "xinput" ["disable", "TPPS/2 IBM TrackPoint"] | ||
| 621 | |||
| 622 | isDisabled :: String -> X Bool | ||
| 623 | isDisabled str = do | ||
| 624 | out <- runProcessWithInput "xinput" ["list", str] "" | ||
| 625 | return $ "disabled" `isInfixOf` out | ||
| 626 | |||
| 627 | |||
| 628 | spawnKeychain :: X () | ||
| 629 | spawnKeychain = do | ||
| 630 | home <- liftIO getHomeDirectory | ||
| 631 | let keys = (map ((home </>) . (".ssh/" ++)) ["id", "id-rsa"]) ++ ["6B13AA67"] | ||
| 632 | liftIO (maybe (return ()) (setEnv "SSH_ASKPASS") =<< findAskpass) | ||
| 633 | safeSpawn "keychain" . (["--agents", "gpg,ssh"] ++)=<< liftIO (filterM doesFileExist keys) | ||
| 634 | where | ||
| 635 | findAskpass = filter `liftM` readFile "/etc/zshrc" | ||
| 636 | filter = listToMaybe . catMaybes . map (stripPrefix "export SSH_ASKPASS=") . lines | ||
| 637 | |||
| 638 | assimilateKeychain :: X () | ||
| 639 | assimilateKeychain = liftIO $ assimilateKeychain' >> return () | ||
| 640 | assimilateKeychain' = tryIOError $ do | ||
| 641 | -- pid <- getProcessID | ||
| 642 | -- tmpDir <- lookupEnv "TMPDIR" | ||
| 643 | -- let tmpDir' = fromMaybe "/tmp" tmpDir | ||
| 644 | -- tmpFile = tmpDir' </> "xmonad-keychain" ++ (show pid) ++ ".env" | ||
| 645 | env <- runProcessWithInput "sh" ["-c", "eval $(keychain --eval --noask --agents gpg,ssh); env"] "" -- > " ++ tmpFile] "" | ||
| 646 | -- env <- readFile tmpFile | ||
| 647 | let envVars = Map.fromList $ map (\(k, v) -> (k, tail' v)) $ map (span (/= '=')) $ envLines | ||
| 648 | envVars' = Map.filterWithKey (\k _ -> k `elem` transfer) envVars | ||
| 649 | transfer = ["SSH_AUTH_SOCK", "SSH_AGENT_PID", "GPG_AGENT_INFO"] | ||
| 650 | envLines = filter (elem '=') $ lines env :: [String] | ||
| 651 | sequence $ map (\(k, c) -> setEnv k c) $ Map.toList envVars' | ||
| 652 | -- removeFile tmpFile | ||
| 653 | where | ||
| 654 | tail' [] = [] | ||
| 655 | tail' (x:xs) = xs | ||
| 656 | |||
| 657 | |||
| 658 | numKeys = [xK_parenleft, xK_parenright, xK_braceright, xK_plus, xK_braceleft, xK_bracketright, xK_bracketleft, xK_exclam, xK_equal, xK_asterisk] | ||
| 659 | |||
| 660 | instance Shrinker CustomShrink where | ||
| 661 | shrinkIt _ "" = [""] | ||
| 662 | shrinkIt s cs | ||
| 663 | | length cs >= 4 = cs : shrinkIt s ((reverse . drop 4 . reverse $ cs) ++ "...") | ||
| 664 | | otherwise = cs : shrinkIt s (init cs) | ||
| 665 | |||
| 666 | xPConfig :: XPConfig | ||
| 667 | xPConfig = def | ||
| 668 | { font = "xft:Fira Mono for Powerline:style=Medium:pixelsize=22.5" | ||
| 669 | , height = 32 | ||
| 670 | , bgColor = "black" | ||
| 671 | , fgColor = "grey" | ||
| 672 | , fgHLight = "green" | ||
| 673 | , bgHLight = "black" | ||
| 674 | , borderColor = "grey" | ||
| 675 | , searchPredicate = (\needle haystack -> all (`isInfixOf` map toLower haystack) . map (map toLower) $ words needle) | ||
| 676 | , position = Top | ||
| 677 | } | ||
| 678 | |||
| 679 | sshOverrides = map (\h -> mkOverride { oHost = h, oCommand = moshCmd . inTmux } ) | ||
| 680 | [ | ||
| 681 | "odin", "odin.asgard.yggdrasil" | ||
| 682 | , "ymir", "ymir.yggdrasil.li", "ymir.niflheim.yggdrasil" | ||
| 683 | , "surtr", "yggdrasil.li", "surtr.yggdrasil.li", "praseodym.org", "surtr.praseodym.org", "surtr.141.li", "141.li" | ||
| 684 | , "vindler", "vindler.alfheim.yggdrasil" | ||
| 685 | , "ullr" | ||
| 686 | , "heimdallr", "heimdallr.asgard.yggdrasil" | ||
| 687 | , "testworx" | ||
| 688 | ] | ||
| 689 | ++ | ||
| 690 | map (\h -> mkOverride { oHost = h, oCommand = moshCmd' "/run/current-system/sw/bin/mosh-server" . withEnv [("TERM", "xterm")] . inTmux} ) | ||
| 691 | [ "bragi", "bragi.asgard.yggdrasil" | ||
| 692 | ] | ||
| 693 | ++ | ||
| 694 | map (\h -> mkOverride { oHost = h, oCommand = sshCmd . withEnv [("TERM", "xterm")] . inTmux } ) | ||
| 695 | [ "remote.cip.ifi.lmu.de" | ||
| 696 | , "uniworx3", "uniworx4", "uniworxdb" | ||
| 697 | ] | ||
| 698 | |||
| 699 | backlight :: (Rational -> Rational) -> X () | ||
| 700 | backlight f = void . xfork . liftIO $ do | ||
| 701 | [ _device | ||
| 702 | , _class | ||
| 703 | , read . Text.unpack -> currentBright | ||
| 704 | , _currentPercentage | ||
| 705 | , read . Text.unpack -> maximumBright | ||
| 706 | ] <- Text.splitOn "," . Text.pack <$> readProcess "brightnessctl" ["-m"] "" | ||
| 707 | let current = currentBright % maximumBright | ||
| 708 | new' = f current * fromIntegral maximumBright | ||
| 709 | new :: Integer | ||
| 710 | new | floor new' < 0 = 0 | ||
| 711 | | ceiling new' > maximumBright = maximumBright | ||
| 712 | | new' >= maximumBright % 2 = ceiling new' | ||
| 713 | | otherwise = floor new' | ||
| 714 | callProcess "brightnessctl" ["-m", "s", show new] | ||
| 715 | |||
| 716 | cycleThrough :: [Rational] -> (Rational -> Rational) | ||
| 717 | cycleThrough opts current = fromMaybe currentOpt $ listToMaybe next' | ||
| 718 | where currentOpt = minimumBy (comparing $ abs . subtract current) opts | ||
| 719 | (_, _ : next') = break (== currentOpt) opts | ||
| 720 | |||
| 721 | cycleKbLayout :: [(String, Maybe String)] -> X () | ||
| 722 | cycleKbLayout [] = return () | ||
| 723 | cycleKbLayout layouts = liftIO $ do | ||
| 724 | next <- (getNext . extract) `liftM` runProcessWithInput "setxkbmap" ["-query"] "" | ||
| 725 | let | ||
| 726 | args = case next of | ||
| 727 | (l, Just v) -> [l, v] | ||
| 728 | (l, Nothing) -> [l] | ||
| 729 | safeSpawn "setxkbmap" args | ||
| 730 | where | ||
| 731 | extract :: String -> Maybe (String, Maybe String) | ||
| 732 | extract str = listToMaybe $ do | ||
| 733 | ["layout:", l] <- str' | ||
| 734 | [(l, Just v) | ["variant:", v] <- str'] ++ pure (l, Nothing) | ||
| 735 | where | ||
| 736 | str' = map words $ lines str | ||
| 737 | getNext :: Maybe (String, Maybe String) -> (String, Maybe String) | ||
| 738 | getNext = maybe (head layouts) getNext' | ||
| 739 | getNext' x = case elemIndex x layouts of | ||
| 740 | Nothing -> getNext Nothing | ||
| 741 | Just i -> layouts !! ((i + 1) `mod` length layouts) | ||
| 742 | |||
| 743 | mpvAll' :: MpvCommand -> IO [MpvResponse] | ||
| 744 | mpvAll' = mpvAll "/var/media/.mpv-ipc" | ||
| 745 | |||
| 746 | mpvOne' :: MpvCommand -> IO (Maybe MpvResponse) | ||
| 747 | mpvOne' = mpvOne "/var/media/.mpv-ipc" | ||
| 748 | |||
| 749 | mediaMpv :: MpvCommand -> X () | ||
| 750 | mediaMpv cmd = void . xfork $ print =<< mpvAll' cmd | ||
| 751 | |||
| 752 | mediaMpvTogglePause :: X () | ||
| 753 | mediaMpvTogglePause = void . xfork $ do | ||
| 754 | paused <- mapM mpvResponse <=< mpvAll' $ MpvGetProperty "pause" | ||
| 755 | if | ||
| 756 | | and paused -> print <=< mpvAll' $ MpvSetProperty "pause" False | ||
| 757 | | otherwise -> print <=< mpvOne' $ MpvSetProperty "pause" True | ||
| 758 | |||
| 759 | myKeys' conf host = Map.fromList $ | ||
| 760 | -- launch a terminal | ||
| 761 | [ ((modm, xK_Return), spawn $ (XMonad.terminal conf) ++ " -e tmux") | ||
| 762 | , ((modm .|. shiftMask, xK_Return), spawn $ XMonad.terminal conf) | ||
| 763 | |||
| 764 | -- launch dmenu | ||
| 765 | --, ((modm, xK_d ), spawn "exe=`dmenu_path | dmenu` && eval \"exec $exe\"") | ||
| 766 | , ((modm, xK_d ), shellPrompt "Run: " xPConfig) | ||
| 767 | , ((modm .|. shiftMask, xK_d ), prompt "Run in Terminal: " ("urxvtc" ++ " -e") xPConfig) | ||
| 768 | , ((modm, xK_at ), sshPrompt sshOverrides xPConfig) | ||
| 769 | |||
| 770 | -- close focused window | ||
| 771 | , ((modm .|. shiftMask, xK_q ), kill) | ||
| 772 | , ((modm .|. controlMask .|. shiftMask, xK_q ), spawn "xkill") | ||
| 773 | |||
| 774 | -- Rotate through the available layout algorithms | ||
| 775 | , ((modm, xK_space ), sendMessage NextLayout) | ||
| 776 | |||
| 777 | -- Reset the layouts on the current workspace to default | ||
| 778 | , ((modm .|. controlMask, xK_r ), (setLayout $ XMonad.layoutHook conf) >> refresh) | ||
| 779 | |||
| 780 | -- Resize viewed windows to the correct size | ||
| 781 | , ((modm, xK_r ), refresh) | ||
| 782 | |||
| 783 | -- Move focus to the next window | ||
| 784 | , ((modm, xK_t ), windows W.focusDown) | ||
| 785 | |||
| 786 | -- Move focus to the previous window | ||
| 787 | , ((modm, xK_n ), windows W.focusUp ) | ||
| 788 | |||
| 789 | -- Move focus to the master window | ||
| 790 | , ((modm, xK_m ), windows W.focusMaster ) | ||
| 791 | |||
| 792 | -- Swap the focused window and the master window | ||
| 793 | , ((modm .|. shiftMask, xK_m ), windows W.swapMaster) | ||
| 794 | |||
| 795 | -- Swap the focused window with the next window | ||
| 796 | , ((modm .|. shiftMask, xK_t ), windows W.swapDown ) | ||
| 797 | |||
| 798 | -- Swap the focused window with the previous window | ||
| 799 | , ((modm .|. shiftMask, xK_n ), windows W.swapUp ) | ||
| 800 | |||
| 801 | -- Swap the focused window with the previous window | ||
| 802 | , ((modm .|. shiftMask .|. controlMask, xK_m), sendMessage SwapWindow) | ||
| 803 | |||
| 804 | , ((modm, xK_Right), sendMessage $ Go R) | ||
| 805 | , ((modm, xK_Left ), sendMessage $ Go L) | ||
| 806 | , ((modm, xK_Up ), sendMessage $ Go U) | ||
| 807 | , ((modm, xK_Down ), sendMessage $ Go D) | ||
| 808 | , ((modm .|. shiftMask , xK_Right), sendMessage $ Move R) | ||
| 809 | , ((modm .|. shiftMask , xK_Left ), sendMessage $ Move L) | ||
| 810 | , ((modm .|. shiftMask , xK_Up ), sendMessage $ Move U) | ||
| 811 | , ((modm .|. shiftMask , xK_Down ), sendMessage $ Move D) | ||
| 812 | -- , ((modm .|. controlMask, xK_Right), withFocused $ keysMoveWindow (10, 0)) | ||
| 813 | -- , ((modm .|. controlMask, xK_Left ), withFocused $ keysMoveWindow (-10, 0)) | ||
| 814 | -- , ((modm .|. controlMask, xK_Up ), withFocused $ keysMoveWindow (0, -10)) | ||
| 815 | -- , ((modm .|. controlMask, xK_Down ), withFocused $ keysMoveWindow (0, 10)) | ||
| 816 | -- Shrink the master area | ||
| 817 | , ((modm, xK_h ), sendMessage Shrink) | ||
| 818 | |||
| 819 | -- Expand the master area | ||
| 820 | , ((modm, xK_s ), sendMessage Expand) | ||
| 821 | |||
| 822 | -- Push window back into tiling | ||
| 823 | , ((modm .|. shiftMask, xK_space ), withFocused $ windows . W.sink) | ||
| 824 | , ((modm, xK_BackSpace), focusUrgent) | ||
| 825 | , ((modm .|. shiftMask, xK_BackSpace), clearUrgents) | ||
| 826 | |||
| 827 | -- Increment the number of windows in the master area | ||
| 828 | , ((modm , xK_comma ), sendMessage (IncMasterN 1)) | ||
| 829 | |||
| 830 | -- Deincrement the number of windows in the master area | ||
| 831 | , ((modm , xK_period), sendMessage (IncMasterN (-1))) | ||
| 832 | |||
| 833 | , ((0, xF86XK_AudioRaiseVolume), safeSpawn "pulseaudio-ctl" ["up", "2"]) | ||
| 834 | , ((0, xF86XK_AudioLowerVolume), safeSpawn "pulseaudio-ctl" ["down", "2"]) | ||
| 835 | , ((0, xF86XK_AudioMute), safeSpawn "pulseaudio-ctl" ["mute"]) | ||
| 836 | , ((0, xF86XK_AudioPause), mediaMpv $ MpvSetProperty "pause" False) | ||
| 837 | , ((0, {-xF86XK_AudioMicMute-} 269025202), safeSpawn "pulseaudio-ctl" ["mute-input"]) | ||
| 838 | , ((0, xF86XK_AudioPlay), mediaMpvTogglePause) | ||
| 839 | , ((modm .|. mod1Mask, xK_space), mediaMpvTogglePause) | ||
| 840 | |||
| 841 | , ((0, xF86XK_MonBrightnessDown), backlight (subtract 5)) | ||
| 842 | , ((0, xF86XK_MonBrightnessUp), backlight (+ 5)) | ||
| 843 | |||
| 844 | , ((modm , xK_Escape), cycleKbLayout (hKbLayouts host)) | ||
| 845 | , ((modm .|. controlMask, xK_Escape), safeSpawn "setxkbmap" $ fst (head $ hKbLayouts host) : maybeToList (snd . head $ hKbLayouts host)) | ||
| 846 | |||
| 847 | -- Toggle the status bar gap | ||
| 848 | -- Use this binding with avoidStruts from Hooks.ManageDocks. | ||
| 849 | -- See also the statusBar function from Hooks.DynamicLog. | ||
| 850 | -- | ||
| 851 | , ((modm , xK_b ), sendMessage ToggleStruts) | ||
| 852 | |||
| 853 | , ((modm .|. shiftMask, xK_p ), safeSpawn "playerctl" ["-a", "pause"]) | ||
| 854 | |||
| 855 | -- Quit xmonad | ||
| 856 | , ((modm .|. shiftMask, xK_e ), io (exitWith ExitSuccess)) | ||
| 857 | |||
| 858 | -- Restart xmonad | ||
| 859 | -- , ((modm .|. shiftMask .|. controlMask, xK_r ), void . xfork $ recompile False >>= flip when (safeSpawn "xmonad" ["--restart"])) | ||
| 860 | , ((modm .|. shiftMask, xK_r ), void . liftIO $ executeFile "xmonad" True [] Nothing) | ||
| 861 | , ((modm .|. shiftMask, xK_l ), void . xfork $ do | ||
| 862 | sessId <- getEnv "XDG_SESSION_ID" | ||
| 863 | safeSpawn "loginctl" ["lock-session", sessId] | ||
| 864 | ) | ||
| 865 | , ((modm .|. shiftMask, xK_s ), safeSpawn "systemctl" ["suspend"]) | ||
| 866 | , ((modm .|. shiftMask, xK_h ), safeSpawn "systemctl" ["hibernate"]) | ||
| 867 | , ((modm .|. shiftMask, xK_b ), backlight $ cycleThrough [1, 3 % 4, 1 % 2, 1 % 4, 1 % 10, 1 % 100, 0] | ||
| 868 | ) | ||
| 869 | , ((modm .|. shiftMask .|. controlMask, xK_b), backlight $ cycleThrough [0, 1 % 100, 1 % 10, 1 % 4, 1 % 2, 3 % 4, 1] | ||
| 870 | ) | ||
| 871 | , ((modm, xK_v ), windows copyToAll) -- @@ Make focused window always visible | ||
| 872 | , ((modm .|. shiftMask, xK_v ), killAllOtherCopies) -- @@ Toggle window state back | ||
| 873 | , ((modm .|. shiftMask, xK_g ), windowPrompt xPConfig Goto wsWindows) | ||
| 874 | , ((modm .|. shiftMask .|. controlMask, xK_g ), windowPrompt xPConfig Bring allWindows) | ||
| 875 | ] | ||
| 876 | ++ | ||
| 877 | |||
| 878 | -- | ||
| 879 | -- mod-[1..9], Switch to workspace N | ||
| 880 | -- | ||
| 881 | -- mod-[1..9], Switch to workspace N | ||
| 882 | -- mod-shift-[1..9], Move client to workspace N | ||
| 883 | -- | ||
| 884 | [((m .|. modm, k), windows $ f i) | ||
| 885 | | (i, k) <- zip (XMonad.workspaces conf) $ numKeys | ||
| 886 | , (f, m) <- [(W.greedyView, 0), (W.shift, shiftMask)] | ||
| 887 | ] | ||
| 888 | ++ | ||
| 889 | [((m .|. modm .|. controlMask, k), void . runMaybeT $ | ||
| 890 | MaybeT (P.getScreen def i) >>= MaybeT . screenWorkspace >>= lift . windows . f | ||
| 891 | ) | ||
| 892 | | (i, k) <- zip (hScreens host) [xK_g, xK_c, xK_r, xK_l] | ||
| 893 | , (f, m) <- [(W.view, 0), (W.shift, shiftMask)] | ||
| 894 | ] | ||
| 895 | where | ||
| 896 | modm = XMonad.modMask conf | ||
| 897 | |||
| 898 | |||
