#!/usr/bin/env python3
#
#    bl-reload-gtk23: Make GTK2/3 reload settings file changes
#    Copyright (C) 2020 2ion <twoion@bunsenlabs.org>
#
#    This program is free software: you can redistribute it and/or modify
#    it under the terms of the GNU General Public License as published by
#    the Free Software Foundation, either version 3 of the License, or
#    (at your option) any later version.
#
#    This program is distributed in the hope that it will be useful,
#    but WITHOUT ANY WARRANTY; without even the implied warranty of
#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
#    GNU General Public License for more details.
#
#    You should have received a copy of the GNU General Public License
#    along with this program.  If not, see <http://www.gnu.org/licenses/>.

from argparse import ArgumentParser, Namespace
import Xlib.display # type: ignore
import Xlib.protocol # type: ignore
import logging
import os
import psutil # type: ignore
import shutil
import signal
import subprocess
import sys

DESCRIPTION = ("""
After changing GTK2 and GTK3 configuration files, notify running GTK2 and GTK3
clients to apply those changes. The notification mechanism used by GTK3 requires
that xsettingsd is running. If it is installed and not running, the program will
launch it. Note that xsettingsd does not read settings from GTK3's settings.ini
- if information is changed, it must be changed in xsettingsd config files as well.

EXIT CODES:
   0 - both gtk2 and gtk3 clients notified successfully
   1 - failed to notify gtk2 clients
   2 - failed to notify gtk3 clients
   3 - failed to notify gtk2 and gtk3 clients

""")

LOG_FORMAT = "%(asctime)s %(levelname)s %(module)s %(funcName)s() : %(message)s"

def getopts() -> Namespace:
    ap = ArgumentParser(description=DESCRIPTION)
    ap.add_argument("-d", "--debug", action="store_true", default=False, help="print debug information")
    ap.add_argument("-f", "--force", action="store_true", default=False, help="ignore all errors")
    opts = ap.parse_args()
    if opts.debug:
        logging.basicConfig(level=logging.DEBUG, format=LOG_FORMAT)
    else:
        logging.basicConfig(level=logging.WARN, format=LOG_FORMAT)
    return opts

def sync_gtk2() -> None:
   """ Tell GTK2 X11 clients to reload the GTK RC files and update their
   appearance/settings if required. This implements this process without GTK/GDK
   in order to be able to drop the dependency on the obsolete pygtk library.
   GTK3/pygobject does not support GTK2.

   This function will always fail on non-X11 platforms as the GTK2 client
   notification mechanism is based on X.

   This implementation is based on the following resources:
      * From libgtk2 2.24.18:
         * gdk_event_send_client_message_to_all_recurse()
         * gdk_screen_broadcast_client_message()
      * From kde-gtk-config https://github.com/KDE/kde-gtk-config/blob/a5d4ddb3b1a27ec2ee4e1b6957a98a57ad56d39c/gtkproxies/reload.c
   """
   display = Xlib.display.Display(display=os.getenv("DISPLAY"))
   wm_state_atom = display.intern_atom("WM_STATE", False)
   gtkrc_atom = display.intern_atom("_GTK_READ_RCFILES", False)

   def send_event(window) -> bool:
      """ Send a _GTK_READ_RCFILES client message to the given X window.
      Returns true unless an exception occurs. """
      window.send_event(
         Xlib.protocol.event.ClientMessage(
            window      = window,
            client_type = gtkrc_atom,
            data        = (8, b"\0" * 20),
         ),
         propagate = 0,
         event_mask = 0
      )
      return True

   def recurse_windows(window, parents) -> bool:
      """ Given a X window, recurse over all its children and selectively
      apply the send_event function to them. Returns true if an event got sent
      to at least one window equal or below the given one. """
      sent = False
      wm_state = window.get_property(wm_state_atom, wm_state_atom, 0, 0)
      name = window.get_wm_name()
      level = len(parents)
      if wm_state is not None:
         sent = send_event(window)
      else:
         tree = window.query_tree()
         for child in tree.children:
            if not recurse_windows(child, parents + [window.id]) and level == 1:
               sent = send_event(window)
      logging.debug("%10s %s %s [%24s] [%s]",
            hex(window.id),
            "W" if not not wm_state else " ",
            "S" if sent else " ",
            name[:24] if name else "",
            ",".join(map(hex, parents)))
      return sent

   for sno in range(0, display.screen_count()):
      screen = display.screen(sno)
      recurse_windows(screen.root, [])

def sync_gtk3():
   """ GTK3 applications can be notified of changes to their theming via
   xsettingsd. This requires that the GTK3 theming information has been updated
   in settings.ini as well as the gsettings schema, managed either by a
   standalone xettingsd implementation or the gnome-settings-daemon. As for now,
   we only support reloading `xsettingsd`: Send SIGHUP if we find it running, or
   start it if it is installed and not running.
      * https://github.com/swaywm/sway/wiki/GTK-3-settings-on-Wayland
      * https://github.com/KDE/kde-gtk-config/blob/a5d4ddb3b1a27ec2ee4e1b6957a98a57ad56d39c/kded/configeditor.cpp#L285
   """
   found = False
   for proc in psutil.process_iter():
      if proc.name() == "xsettingsd":
         proc.send_signal(signal.SIGHUP)
         found = True
         logging.debug("Found xsettingsd process and sent SIGHUP: PID %d", proc.pid)
         break
   if not found:
      xsettingsd_binary = shutil.which("xsettingsd")
      if xsettingsd_binary is not None:
         proc = subprocess.Popen([xsettingsd_binary], cwd="/", start_new_session=True)
         logging.debug("xsettingsd not running; started from %s; PID: %d", xsettingsd_binary, proc.pid)
      else:
         raise RuntimeError("xsettingsd not running and not in PATH, no settings changes propagated")

def main() -> int:
   opts = getopts()
   ret = 0
   try:
      sync_gtk2()
   except Exception as err:
      logging.warning("Failed to reload GTK2 settings: %s", err)
      ret |= 1<<0
   try:
      sync_gtk3()
   except Exception as err:
      logging.warning("Failed to reload GTK3 settings: %s", err)
      ret |= 1<<1
   return 0 if opts.force else ret

if __name__ == "__main__":
    sys.exit(main())