#!/usr/bin/env python3 """Simple dimmer for xss-lock. It dim the screen using a provided delay and display a countdown. It will stop itself when the locker window is mapped. """ # It assumes we are using a compositor. import gi gi.require_version("Gtk", "3.0") gi.require_version("Gst", "1.0") from gi.repository import Gtk, Gdk, GLib, GdkPixbuf, GObject, Gst import cairo import argparse import threading import time import math import os import warnings from Xlib import display, X from Xlib.error import BadWindow from Xlib.protocol.event import MapNotify def on_xevent(source, condition, xdisplay, locker): while xdisplay.pending_events(): event = xdisplay.next_event() if event.type != X.MapNotify: continue try: wmclass = event.window.get_wm_class() except BadWindow: continue if wmclass and wmclass[1] == locker: Gtk.main_quit() return False return True def on_realize(widget): window = widget.get_window() window.set_override_redirect(True) def on_draw(widget, event, options, background, start): x, y = widget.get_position() wwidth, wheight = widget.get_size() delta = options.end_opacity - options.start_opacity elapsed = time.monotonic() - start current = easing_functions[options.easing_function](elapsed / options.delay) opacity = delta * current + options.start_opacity cctx = event # Background scale = widget.get_scale_factor() bg = None if background: bg = background.new_subpixbuf( x * scale, y * scale, wwidth * scale, wheight * scale ) cctx.set_operator(cairo.OPERATOR_SOURCE) if not bg: cctx.set_source_rgba(0, 0, 0, opacity) cctx.paint() else: cctx.save() cctx.scale(1 / scale, 1 / scale) Gdk.cairo_set_source_pixbuf(cctx, bg, 0, 0) cctx.paint_with_alpha(opacity) cctx.restore() # Remaining time cctx.set_operator(cairo.OPERATOR_OVER) remaining = str(round(options.delay - elapsed)) cctx.select_font_face(options.font, cairo.FONT_SLANT_NORMAL, cairo.FONT_WEIGHT_BOLD) cctx.set_font_size(wheight // 4) _, _, twidth, theight, _, _ = cctx.text_extents("8" * len(remaining)) text_position = wwidth // 2 - twidth // 2, wheight // 2 + theight // 2 cctx.move_to(*text_position) cctx.set_source_rgba(1, 1, 1, opacity) cctx.show_text(remaining) cctx.move_to(*text_position) cctx.set_source_rgba(0, 0, 0, opacity * 2) cctx.set_line_width(4) cctx.text_path(remaining) cctx.stroke() def on_refresh(window, options, start): window.queue_draw() elapsed = time.monotonic() - start if elapsed < options.delay: next_step = min(options.step, options.delay - elapsed) GLib.timeout_add(next_step * 1000, on_refresh, window, options, start) def on_sound_ticker(options, sound, start): elapsed = time.monotonic() - start if elapsed < options.delay: if sound is None: Gst.init([]) sound = Gst.ElementFactory.make("playbin", "xss-dimmer") sound.set_property("uri", f"file://{os.path.abspath(options.sound)}") sound.set_state(Gst.State.NULL) sound.seek_simple( Gst.Format.TIME, Gst.SeekFlags.FLUSH | Gst.SeekFlags.KEY_UNIT, 0 ) sound.set_state(Gst.State.PLAYING) GLib.timeout_add(1000, on_sound_ticker, options, sound, now) # See: https://easings.net/ easing_functions = { "none": lambda x: x, "out-circ": lambda x: math.sqrt(1 - pow(x - 1, 2)), "out-sine": lambda x: math.sin(x * math.pi / 2), "out-cubic": lambda x: 1 - pow(1 - x, 3), "out-quint": lambda x: 1 - pow(1 - x, 5), "out-expo": lambda x: 1 - pow(2, -10 * x), "out-quad": lambda x: 1 - (1 - x) * (1 - x), "out-bounce": ( lambda n1, d1: lambda x: n1 * x * x if x < 1 / d1 else n1 * pow(x - 1.5 / d1, 2) + 0.75 if x < 2 / d1 else n1 * pow(x - 2.25 / d1, 2) + 0.9375 if (x < 2.5 / d1) else n1 * pow(x - 2.625 / d1, 2) + 0.984375 )(7.5625, 2.75), "out-elastic": ( lambda x: pow(2, -10 * x) * math.sin((x * 10 - 0.75) * (2 * math.pi) / 3) + 1 ), "inout-quad": lambda x: 2 * x * x if x < 0.5 else 1 - pow(-2 * x + 2, 2) / 2, "inout-quart": ( lambda x: 8 * x * x * x * x if x < 0.5 else 1 - pow(-2 * x + 2, 4) / 2 ), "inout-expo": ( lambda x: pow(2, 20 * x - 10) / 2 if x < 0.5 else (2 - pow(2, -20 * x + 10)) / 2 ), "inout-bounce": ( lambda x: (1 - easing_functions["out-bounce"](1 - 2 * x)) / 2 if x < 0.5 else (1 + easing_functions["out-bounce"](2 * x - 1)) / 2 ), } if __name__ == "__main__": now = time.monotonic() parser = argparse.ArgumentParser() add = parser.add_argument add("--start-opacity", type=float, default=0, help="initial opacity") add("--end-opacity", type=float, default=1, help="final opacity") add("--step", type=float, default=0.1, help="step for changing opacity") add("--delay", type=float, default=10, help="delay from start to end") add("--font", default="Iosevka Aile", help="font for countdown") add("--locker", default="xsecurelock", help="quit if window class detected") add("--background", help="use a background instead of black") add("--no-randr", help="disable RandR", action="store_true") add( "--easing-function", default="none", choices=easing_functions.keys(), help="easing function for opacity", ) add("--sound", help="play a sound for each second elapsed while dimmer running") options = parser.parse_args() # This is a hack! try: with open( os.path.join(os.environ["XDG_RUNTIME_DIR"], "i3", "outputs.txt") ) as f: if int(f.read()) == 1: options.no_randr = True except: pass background = None if options.background: try: background = GdkPixbuf.Pixbuf.new_from_file(options.background) except Exception: pass # Setup dimmer windows on each monitor gdisplay = Gdk.Display.get_default() geoms = [] if options.no_randr: with warnings.catch_warnings(): warnings.filterwarnings("ignore", category=DeprecationWarning) screen = gdisplay.get_screen(0) geoms.append((0, 0, screen.get_width(), screen.get_height())) else: for i in range(gdisplay.get_n_monitors()): geom = gdisplay.get_monitor(i).get_geometry() geoms.append((geom.x, geom.y, geom.width, geom.height)) for x, y, width, height in geoms: window = Gtk.Window() window.set_app_paintable(True) window.set_type_hint(Gdk.WindowTypeHint.SPLASHSCREEN) window.set_visual(window.get_screen().get_rgba_visual()) window.set_default_size(width, height) window.move(x, y) window.connect("draw", on_draw, options, background, now) window.connect("delete-event", Gtk.main_quit) window.connect("realize", on_realize) window.show_all() # Schedule refresh with window.queue_draw() on_refresh(window, options, now) # Watch for locker window xdisplay = display.Display() root = xdisplay.screen().root root.change_attributes(event_mask=X.SubstructureNotifyMask) channel = GLib.IOChannel.unix_new(xdisplay.fileno()) channel.set_encoding(None) channel.set_buffered(False) GLib.io_add_watch( channel, GLib.PRIORITY_DEFAULT, GLib.IOCondition.IN, on_xevent, xdisplay, options.locker, ) xdisplay.pending_events() # otherwise, socket is inactive if options.sound: GLib.timeout_add(options.delay * 1000 // 3, on_sound_ticker, options, None, now) # Main loop Gtk.main()