diff options
-rw-r--r-- | XMonad/Hooks/UrgencyHook.hs | 67 |
1 files changed, 57 insertions, 10 deletions
diff --git a/XMonad/Hooks/UrgencyHook.hs b/XMonad/Hooks/UrgencyHook.hs index 82c20fe..7ef09b5 100644 --- a/XMonad/Hooks/UrgencyHook.hs +++ b/XMonad/Hooks/UrgencyHook.hs @@ -78,14 +78,16 @@ import XMonad.Util.Dzen (dzenWithArgs, seconds) import qualified XMonad.Util.ExtensibleState as XS import XMonad.Util.NamedWindows (getName) import XMonad.Util.Timer (TimerId, startTimer, handleTimer) +import XMonad.Util.WindowProperties (getProp32) import Control.Applicative ((<$>)) import Control.Monad (when) import Data.Bits (testBit) import Data.List (delete, (\\)) -import Data.Maybe (listToMaybe, maybeToList) +import Data.Maybe (listToMaybe, maybeToList, fromMaybe) import qualified Data.Set as S import System.IO (hPutStrLn, stderr) +import Foreign.C.Types (CLong) -- $usage -- @@ -310,12 +312,34 @@ readReminders = XS.get adjustReminders :: ([Reminder] -> [Reminder]) -> X () adjustReminders = XS.modify -clearUrgency :: Window -> X () -clearUrgency w = adjustUrgents (delete w) >> adjustReminders (filter $ (w /=) . window) data WithUrgencyHook h = WithUrgencyHook h UrgencyConfig deriving (Read, Show) +-- | Change the _NET_WM_STATE property by applying a function to the list of atoms. +changeNetWMState :: Display -> Window -> ([CLong] -> [CLong]) -> X () +changeNetWMState dpy w f = do + wmstate <- getAtom "_NET_WM_STATE" + wstate <- fromMaybe [] `fmap` getProp32 wmstate w + let ptype = 4 -- atom property type for changeProperty + io $ changeProperty32 dpy w wmstate ptype propModeReplace (f wstate) + return () + +-- | Add an atom to the _NET_WM_STATE property. +addNetWMState :: Display -> Window -> Atom -> X () +addNetWMState dpy w atom = changeNetWMState dpy w $ ((fromIntegral atom):) + +-- | Remove an atom from the _NET_WM_STATE property. +removeNetWMState :: Display -> Window -> Atom -> X () +removeNetWMState dpy w atom = changeNetWMState dpy w $ delete (fromIntegral atom) + +-- | Get the _NET_WM_STATE propertly as a [CLong] +getNetWMState :: Window -> X [CLong] +getNetWMState w = do + a_wmstate <- getAtom "_NET_WM_STATE" + fromMaybe [] `fmap` getProp32 a_wmstate w + + -- The Non-ICCCM Manifesto: -- Note: Some non-standard choices have been made in this implementation to -- account for the fact that things are different in a tiling window manager: @@ -331,20 +355,40 @@ data WithUrgencyHook h = WithUrgencyHook h UrgencyConfig handleEvent :: UrgencyHook h => WithUrgencyHook h -> Event -> X () handleEvent wuh event = case event of + -- WM_HINTS urgency flag PropertyEvent { ev_event_type = t, ev_atom = a, ev_window = w } -> do when (t == propertyNotify && a == wM_HINTS) $ withDisplay $ \dpy -> do WMHints { wmh_flags = flags } <- io $ getWMHints dpy w - if (testBit flags urgencyHintBit) then do - adjustUrgents (\ws -> if elem w ws then ws else w : ws) - callUrgencyHook wuh w - else - clearUrgency w - userCodeDef () =<< asks (logHook . config) + if (testBit flags urgencyHintBit) then markUrgent w else markNotUrgent w + -- Window destroyed DestroyWindowEvent {ev_window = w} -> - clearUrgency w + markNotUrgent w + -- _NET_WM_STATE_DEMANDS_ATTENTION requested by client + ClientMessageEvent {ev_event_display = dpy, ev_window = w, ev_message_type = t, ev_data = action:atoms} -> do + a_wmstate <- getAtom "_NET_WM_STATE" + a_da <- getAtom "_NET_WM_STATE_DEMANDS_ATTENTION" + wstate <- getNetWMState w + let demandsAttention = fromIntegral a_da `elem` wstate + remove = 0 + add = 1 + toggle = 2 + when (t == a_wmstate && fromIntegral a_da `elem` atoms) $ do + when (action == add || (action == toggle && not demandsAttention)) $ do + markUrgent w + addNetWMState dpy w a_da + when (action == remove || (action == toggle && demandsAttention)) $ do + markNotUrgent w + removeNetWMState dpy w a_da _ -> mapM_ handleReminder =<< readReminders where handleReminder reminder = handleTimer (timer reminder) event $ reminderHook wuh reminder + markUrgent w = do + adjustUrgents (\ws -> if elem w ws then ws else w : ws) + callUrgencyHook wuh w + userCodeDef () =<< asks (logHook . config) + markNotUrgent w = do + adjustUrgents (delete w) >> adjustReminders (filter $ (w /=) . window) + userCodeDef () =<< asks (logHook . config) callUrgencyHook :: UrgencyHook h => WithUrgencyHook h -> Window -> X () callUrgencyHook (WithUrgencyHook hook UrgencyConfig { suppressWhen = sw, remindWhen = rw }) w = @@ -378,6 +422,9 @@ shouldSuppress sw w = elem w <$> suppressibleWindows sw cleanupUrgents :: SuppressWhen -> X () cleanupUrgents sw = do sw' <- suppressibleWindows sw + a_da <- getAtom "_NET_WM_STATE_DEMANDS_ATTENTION" + dpy <- withDisplay (\dpy -> return dpy) + mapM_ (\w -> removeNetWMState dpy w a_da) sw' adjustUrgents (\\ sw') >> adjustReminders (filter $ ((`notElem` sw') . window)) suppressibleWindows :: SuppressWhen -> X [Window] |