Skip to content

Commit

Permalink
Add driver/XMonad.hs
Browse files Browse the repository at this point in the history
  • Loading branch information
sol committed Dec 19, 2024
1 parent 8b7a6fd commit 8910bda
Show file tree
Hide file tree
Showing 3 changed files with 87 additions and 1 deletion.
80 changes: 80 additions & 0 deletions driver/XMonad.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,80 @@
{-# LANGUAGE BlockArguments #-}
{-# LANGUAGE NoOverloadedStrings #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE ViewPatterns #-}
module XMonad where

-- import Control.Exception (IOException, try)
import System.Process
import System.Posix.Process (getProcessID)
-- import Data.Maybe (listToMaybe)
import Data.Functor
import System.Exit
import Data.List

import Imports

newtype PID = PID String
newtype XWindowId = XWindowId String
newtype Tag = Tag String

instance IsString Tag where
fromString = Tag

-- Get the parent PID of a process
getParentPid :: PID -> IO (Maybe PID)
getParentPid (PID pid) = readProcessWithExitCode "ps" ["-o", "ppid=", "-p", pid] "" <&> \ case
(_, result, _) -> case strip result of
"" -> Nothing
ppid -> Just (PID ppid)

getWindowIdForPid :: PID -> IO (Maybe XWindowId)
getWindowIdForPid (PID pid) = do
windowIds <- filter (isPrefixOf "0x") . words <$> readProcess "xprop" ["-root", "_NET_CLIENT_LIST"] ""
findWindow windowIds
where
findWindow :: [String] -> IO (Maybe XWindowId)
findWindow = \ case
[] -> return Nothing
wid : rest -> do
wmPidOutput <- readProcess "xprop" ["-id", wid, "_NET_WM_PID"] ""

case listToMaybe (filter (all (`elem` "0123456789")) (words wmPidOutput)) of
Just wmPid | wmPid == pid -> return (Just $ XWindowId wid)
_ -> findWindow rest

-- Recursively find the first ancestor PID that is an X window
findAncestorWindowId :: PID -> IO (Maybe XWindowId)
findAncestorWindowId pid = do
windowId <- getWindowIdForPid pid
case windowId of
Just wid -> return (Just wid)
Nothing -> do
parentPid <- getParentPid pid
case parentPid of
Just ppid -> findAncestorWindowId ppid
Nothing -> return Nothing

addTag :: Tag -> XWindowId -> IO ()
addTag (Tag name) (XWindowId wid) = do
result <- readProcess "xprop" ["-id", wid, "_XMONAD_TAGS"] ""

let
tags :: String
tags = if "not found" `elem` words result then "" else extractTags result

newTags :: String
newTags = if null tags then name else tags <> " " <> name

callProcess "xprop" ["-id", wid, "-f", "_XMONAD_TAGS", "8s", "-set", "_XMONAD_TAGS", newTags]
where
extractTags :: String -> String
extractTags = unwords . drop 1 . words . last . lines

tagSelfWith :: Tag -> IO ()
tagSelfWith name = do
pid <- PID . show <$> getProcessID
result <- findAncestorWindowId pid
case result of
Just wid -> addTag name wid
Nothing -> exitFailure
5 changes: 4 additions & 1 deletion driver/sensei.hs
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,10 @@ module Main (main) where

import System.Environment

import qualified XMonad
import Run

main :: IO ()
main = getArgs >>= run
main = do
XMonad.tagSelfWith "sensei"
getArgs >>= run
3 changes: 3 additions & 0 deletions sensei.cabal

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit 8910bda

Please sign in to comment.