Show HN: A Python Swiss-knife to manage Wayland compositors (Hyprland, Sway)

https://github.com/juienpro/easyland

Easyland

Easyland is a Python framework to manage your wayland compositor (Hyprland, Sway) configuration by reacting to events. With Easyland, you can dismiss many side tools like Kanshi, hypridle, swayidle, etc. and script your environment according to your preferences.

Available listeners

The tool allows to listen for these events and to execute commands in response.

Why this tool?

Good question.

Initially, I was a bit stressed by the number of tools needed with Hyprland (Kanshi & hypridle notably), and also by the number of bugs despite the awesome efforts of the developers.

I wanted to have a deeper control on my system, and to be able to script it as I wanted.

To give an example, my laptop screen brightness was always at 100% when I undock it, and Kanshi does not allow to add shell commands. This is only one small example of the numerous limitations I met during my setup of Hyprland.

By scripting my Desktop in Python, I have more control to implement what I want.

Installation

  1. Install Easyland in a python environment
  1. Copy an example of configuration files from here

  2. Modify it according to your needs

  3. Launch easyland -c <path_to_your_config_file

This program needs the following external tools:

  • The socat binary (Arch)
  • The gdbus binary (Arch)

Depending if you use Hyprland or Sway, you will need hyprctl or swaymsg.

If it's not done automatically, before using PyWayland, you will need to execute pywayland.scanner to generate all protocols:

python -m pywayland.scanner

or

How to use

The easyland package provides the easyland CLI command, which loads your custom Python file configuration with the -c parameter:

easyland -c ~/home/.config/hyprland/myconfig.py

Where myconfig.py contains such a content, explained in details below:

from easyland import logger, command
###############################################################################
# Set active listeners
###############################################################################
listeners = {
    "hyprland": { "socket_path": "/tmp/hypr/$HYPRLAND_INSTANCE_SIGNATURE/.socket2.sock" },
    'systemd_logind': {},
    'idle': {}
}
###############################################################################
# Method executed at start 
###############################################################################
def init():
    set_monitors()
###############################################################################
# Idle configuration
# Format: [timeout in seconds, [commands to run], [commands to run on resume]]
###############################################################################
def idle_config():
    return [
        [150, ['brightnessctl -s set 0'], ['brightnessctl -r']],
        [600, ['pidof hyprlock || hyprlock']],
        [720, ['hyprctl dispatch dpms off'], ['hyprctl dispatch dpms on']]
    ]
###############################################################################
# Handler of Hyprland IPC events
# List of events: https://wiki.hyprland.org/IPC/
###############################################################################
def on_hyprland_event(event, argument):
    if event in [ "monitoradded", "monitorremoved" ]:
        logger.info('Handling hyprland event: ' + event)
        set_monitors()
###############################################################################
# Handlers of Systemd logind events
###############################################################################
def on_PrepareForSleep(payload):
    if 'true' in payload:
        logger.info("Locking the screen before suspend")
        command.exec("pidof hyprlock || hyprlock", True)
# To use this handler, you need to launch your locker (hyprlock or swaylock) like this: hyprlock && loginctl unlock-session
# def on_Unlock():
#     logger.info("Unlocking the screen")
# To use this handler, you need to launch your locker like this: loginctl lock-session
# def on_lock():
#     logger.info("Locking the screen")
###############################################################################
# Various methods
###############################################################################
def set_monitors():
    logger.info('Setting monitors')
    if command.hyprland_get_monitor(description="HP 22es") is not None:
        command.exec('hyprctl keyword monitor "eDP-1,preferred,auto,2"')
        # command.exec('hyprctl keyword monitor "eDP-1,disable"')
    else:
        command.exec('hyprctl keyword monitor "eDP-1,preferred,auto,2"')
        command.exec("brightnessctl -s set 0")

Example of configuration

You can find an example in the config_examples folder. We'll explain step-by-step the configuration called Hyprland.py.

Importing helpers

from easyland import logger, command

Easyland has helper tools to log everything (console and file) and execute commands. Just import the two.

Configure listeners

listeners = {
    "hyprland": { "socket_path": "/tmp/hypr/$HYPRLAND_INSTANCE_SIGNATURE/.socket2.sock" },
    'systemd_logind': {},
    'idle': {}
}

The listeners to launch at the startup. There are currently three listeners:

  • hyprland to listen Hyprland IPC events
  • systemd_logind to monitor for Systemd Logind events
  • idle which allows you to react when your computer has no activity

Each one can take take on or more parameters. See reference below.

Configure Idling

def idle_config():
    return [
        [150, ['brightnessctl -s set 0'], ['brightnessctl -r']],
        [600, ['pidof hyprlock || hyprlock']],
        [720, ['hyprctl dispatch dpms off'], ['hyprctl dispatch dpms on']]
    ]

This method configure the Idle part. It should return the list of your idle actions. Each action has three parameters:

  • The timeout in seconds
  • The list of commands to execute when the timeout occurs
  • The optional list of commands when the timeout is resumed (eg. once there is user activity after the timeout)

Here, at 720 seconds, the screens are turned off. Then, if some activities are detected, they are turned on.

Configure monitors

def on_hyprland_event(event, argument):
    if event in [ "monitoradded", "monitorremoved" ]:
        logger.info('Handling hyprland event: ' + event)
        set_monitors()

The method on_hyprland_event allows you to handle Hyprland IPC events. All events are avalable here.

In this case, when we connect or disconnect a monitor, we call a method to set our monitors according to our preferences. This method is as follows.

def set_monitors(self):
    logger.info('Setting monitors')
    if command.hyprland_get_monitor(description="HP 22es") is not None:
        command.exec('hyprctl keyword monitor "eDP-1,disable"')
    else:
        command.exec('hyprctl keyword monitor "eDP-1,preferred,auto,2"')
        command.exec("brightnessctl -s set 0")

We use the hyprland_get_monitor command helper to get the configuration of a particular monitor. hyprland_get_monitor accepts the name of the monitor, its description, the maker or the model. If the screen is not found, this method returns None.

So, when we detect a "HP 22es" monitor, we disable the screen of the laptop. Otherwise, we turn on the monitor of the laptop, and we put the brightness at the lowest level (in my configuration, when I undock my laptop, the brightness is at 100%)

For Sway, you have the sway_get_monitor helper method.

Configure suspend

def on_PrepareForSleep(payload):
    if 'true' in payload:
        logger.info("Locking the screen before suspend")
        command.exec("pidof hyprlock || hyprlock", True)

The methods on_Whatever(payload) are automatically called when the signal Whatever is sent by Systemd Logind. Here, we are listening for the signal "PrepareForSleep" which is called just before you computer is suspending.

The second parameter of the command.exec helper allows you to execute a command in the background. It's necessary here, otherwhise Easyland will wait indefinitely until the screen is unlocked.

Other tricks

You may be interested to listen for Lock and Unlock events emitted from Systemd when you call loginctl lock-session and loginctl unlock-session. However, keep in mind that hyprlock and swaylock do not send any signal for these events, so you need to hack that.

# To use this handler, you need to launch your locker (hyprlock or swaylock) like this: hyprlock && loginctl unlock-session
def on_Unlock():
    logger.info("Unlocking the screen")

To receive the Systemd Unlock signal, you should launch your screen locker with the following command: hyprlock && loginctl unlock-session, so Systemd will send the Unlock signal when the screen is unlocked.

For locking, keep in mind that hyprlock and swaylock do not listen for the Systemd Lock event, so you need to it manually.

# To use this handler, you need to launch your locker like this: loginctl lock-session
def on_Lock():
     logger.info("Locking the screen")
     command.exec('pidof hyprlock || hyprlock', True)
     # Do other actions if needed

Alternative to on_WhateverSignal method

Alternatively to write several methods to listen for Systemd events, you can also define method on_systemd_event and add a condition to achieve what you want:

def on_systemd_event(sender, signal, payload)
    if signal == 'Lock':
        ...
    if signal == 'PrepareForSleep':
        ...

References

Listeners parameters

Hyprland listener parameters

  • socket_path: The path of the Hyprland IPC socket

Sway listener parameters

  • event_types: The type of events to be listened for

Idle listener parameters

None.

systemd_logind parameters

None.

Listeners handler methods

Sender Handler method to add to your class Arguments
Hyprland on_hyprland_event event, argument
Sway on_sway_event_[type] payload
Systemd Logind on_systemd_event sender, signal, payload
Systemd Logind on_[signal] payload

Hyprland events

They are well documented here.

Sway event types

For Sway, the current event types are those defined in the IPC manual

Systemd Logind events

These events are called "signals" in the Systemd terminology.

They are not well documented but you can try to read that (good luck).

Some examples that can be useful:

Member Description
PrepareForShutdown Sent before a shutdown
PrepareForSleep Sent before suspend
Lock Sent when a lock is requested, eg loginctrl lock-session
Unlock Sent when an unlock is requested
SessionNew When a session is created

Keep in mind that these signals are independent from Wayland/Hyprland/Sway. My recommendation would be to always configure your compositor to use loginctl to send the signals, and add a listener in Easyland to achieve what you want.

Available helpers

Method Usage Arguments
command.exec Execute a command, eventually in the background cmd (string), background (bool, default False), decode_json (bool, default False)
command.hyprland_get_all_monitors Get all monitors and their configuration through Hyprland IPC None
command.hyprland_get_monitor Get the config of one monitor, None if not found name, description, make, model
command.sway_get_all_monitors Get all monitors and their configuration through Hyprland IPC None
command.sway_get_monitor Get the config of one monitor, None if not found name, make, model
logger Log messages to easyland.log and to STDOUT use logger.info, logger.error, for the severity etc.
idle_config Set the idle configuration None

Contributions

  • Integrating other DBUS services should be easy with Easyland (type dbusctl to list all avalable DBUS on your system). Do not hesitate to let me know what you need.
  • Better tests for Sway. I use Hyprland so feel free to submit bugs if you are using Sway and see an issue.

If you see some bugs or propose patches, feel free to contribute.

Thanks

Thanks to the developer(s) of Hyprland for their fantastic compositor. I tried so many ones in the past, and this has been Hyprland that convinced me to do the switch from KDE :-)

{
"by": "julienmaker",
"descendants": 1,
"id": 40245802,
"kids": [
40245803
],
"score": 1,
"time": 1714729101,
"title": "Show HN: A Python Swiss-knife to manage Wayland compositors (Hyprland, Sway)",
"type": "story",
"url": "https://github.com/juienpro/easyland"
}
{
"author": "juienpro",
"date": null,
"description": "A python swiss-knife to manage Wayland compositors like Hyprland and Sway - juienpro/easyland",
"image": "https://opengraph.githubassets.com/a89dbf6caba56aaf851c82620dcd2d1fb9f3f920313adf854125db91ec58d683/juienpro/easyland",
"logo": "https://logo.clearbit.com/github.com",
"publisher": "GitHub",
"title": "GitHub - juienpro/easyland: A python swiss-knife to manage Wayland compositors like Hyprland and Sway",
"url": "https://github.com/juienpro/easyland"
}
{
"url": "https://github.com/juienpro/easyland",
"title": "GitHub - juienpro/easyland: A python swiss-knife to manage Wayland compositors like Hyprland and Sway",
"description": "Easyland Easyland is a Python framework to manage your wayland compositor (Hyprland, Sway) configuration by reacting to events. With Easyland, you can dismiss many side tools like Kanshi, hypridle, swayidle,...",
"links": [
"https://github.com/juienpro/easyland"
],
"image": "https://opengraph.githubassets.com/a89dbf6caba56aaf851c82620dcd2d1fb9f3f920313adf854125db91ec58d683/juienpro/easyland",
"content": "<div><article><p></p><h2>Easyland</h2><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#easyland\"></a><p></p>\n<p><strong>Easyland</strong> is a Python framework to manage your wayland compositor (Hyprland, Sway) configuration by reacting to events. With <strong>Easyland</strong>, you can dismiss many side tools like Kanshi, hypridle, swayidle, etc. and script your environment according to your preferences.</p>\n<p></p><h2>Available listeners</h2><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#available-listeners\"></a><p></p>\n<ul>\n<li><a target=\"_blank\" href=\"https://wiki.hyprland.org/IPC/\">Hyprland IPC</a> event list</li>\n<li><a target=\"_blank\" href=\"https://man.archlinux.org/man/sway-ipc.7.en\">Sway IPC</a></li>\n<li>Systemd signals</li>\n<li>Native Wayland Idle system (ext_idle_notify_v1)</li>\n</ul>\n<p>The tool allows to listen for these events and to execute commands in response.</p>\n<p></p><h2>Why this tool?</h2><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#why-this-tool\"></a><p></p>\n<p>Good question.</p>\n<p>Initially, I was a bit stressed by the number of tools needed with Hyprland (Kanshi &amp; hypridle notably), and also by the number of bugs despite the awesome efforts of the developers.</p>\n<p>I wanted to have a deeper control on my system, and to be able to script it as I wanted.</p>\n<p>To give an example, my laptop screen brightness was always at 100% when I undock it, and Kanshi does not allow to add shell commands. This is only one small example of the numerous limitations I met during my setup of Hyprland.</p>\n<p>By scripting my Desktop in Python, I have more control to implement what I want.</p>\n<p></p><h2>Installation</h2><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#installation\"></a><p></p>\n<ol>\n<li>Install Easyland in a python environment</li>\n</ol>\n<ol>\n<li>\n<p>Copy an example of configuration files from <a target=\"_blank\" href=\"https://github.com/juienpro/easyland/tree/main/config_examples\">here</a></p>\n</li>\n<li>\n<p>Modify it according to your needs</p>\n</li>\n<li>\n<p>Launch <code>easyland -c &lt;path_to_your_config_file</code></p>\n</li>\n</ol>\n<p>This program needs the following external tools:</p>\n<ul>\n<li>The <code>socat</code> binary (<a target=\"_blank\" href=\"https://archlinux.org/packages/extra/x86_64/socat/\">Arch</a>)</li>\n<li>The <code>gdbus</code> binary (<a target=\"_blank\" href=\"https://archlinux.org/packages/core/x86_64/glib2/\">Arch</a>)</li>\n</ul>\n<p>Depending if you use Hyprland or Sway, you will need <code>hyprctl</code> or <code>swaymsg</code>.</p>\n<p>If it's not done automatically, before using PyWayland, you will need to execute <code>pywayland.scanner</code> to generate all protocols:</p>\n<div><pre><code>python -m pywayland.scanner\n</code></pre></div>\n<p>or</p>\n<p></p><h2>How to use</h2><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#how-to-use\"></a><p></p>\n<p>The <strong>easyland</strong> package provides the <code>easyland</code> CLI command, which loads your custom Python file configuration with the -c parameter:</p>\n<div><pre><code>easyland -c ~/home/.config/hyprland/myconfig.py\n</code></pre></div>\n<p>Where <code>myconfig.py</code> contains such a content, explained in details below:</p>\n<div><pre><code>from easyland import logger, command\n###############################################################################\n# Set active listeners\n###############################################################################\nlisteners = {\n \"hyprland\": { \"socket_path\": \"/tmp/hypr/$HYPRLAND_INSTANCE_SIGNATURE/.socket2.sock\" },\n 'systemd_logind': {},\n 'idle': {}\n}\n###############################################################################\n# Method executed at start \n###############################################################################\ndef init():\n set_monitors()\n###############################################################################\n# Idle configuration\n# Format: [timeout in seconds, [commands to run], [commands to run on resume]]\n###############################################################################\ndef idle_config():\n return [\n [150, ['brightnessctl -s set 0'], ['brightnessctl -r']],\n [600, ['pidof hyprlock || hyprlock']],\n [720, ['hyprctl dispatch dpms off'], ['hyprctl dispatch dpms on']]\n ]\n###############################################################################\n# Handler of Hyprland IPC events\n# List of events: https://wiki.hyprland.org/IPC/\n###############################################################################\ndef on_hyprland_event(event, argument):\n if event in [ \"monitoradded\", \"monitorremoved\" ]:\n logger.info('Handling hyprland event: ' + event)\n set_monitors()\n###############################################################################\n# Handlers of Systemd logind events\n###############################################################################\ndef on_PrepareForSleep(payload):\n if 'true' in payload:\n logger.info(\"Locking the screen before suspend\")\n command.exec(\"pidof hyprlock || hyprlock\", True)\n# To use this handler, you need to launch your locker (hyprlock or swaylock) like this: hyprlock &amp;&amp; loginctl unlock-session\n# def on_Unlock():\n# logger.info(\"Unlocking the screen\")\n# To use this handler, you need to launch your locker like this: loginctl lock-session\n# def on_lock():\n# logger.info(\"Locking the screen\")\n###############################################################################\n# Various methods\n###############################################################################\ndef set_monitors():\n logger.info('Setting monitors')\n if command.hyprland_get_monitor(description=\"HP 22es\") is not None:\n command.exec('hyprctl keyword monitor \"eDP-1,preferred,auto,2\"')\n # command.exec('hyprctl keyword monitor \"eDP-1,disable\"')\n else:\n command.exec('hyprctl keyword monitor \"eDP-1,preferred,auto,2\"')\n command.exec(\"brightnessctl -s set 0\")\n</code></pre></div>\n<p></p><h3>Example of configuration</h3><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#example-of-configuration\"></a><p></p>\n<p>You can find an example in the <code>config_examples</code> folder. We'll explain step-by-step the configuration called <code>Hyprland.py</code>.</p>\n<p></p><h4>Importing helpers</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#importing-helpers\"></a><p></p>\n<div><pre><code>from easyland import logger, command\n</code></pre></div>\n<p>Easyland has helper tools to log everything (console and file) and execute commands. Just import the two.</p>\n<p></p><h4>Configure listeners</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#configure-listeners\"></a><p></p>\n<div><pre><code>listeners = {\n \"hyprland\": { \"socket_path\": \"/tmp/hypr/$HYPRLAND_INSTANCE_SIGNATURE/.socket2.sock\" },\n 'systemd_logind': {},\n 'idle': {}\n}\n</code></pre></div>\n<p>The listeners to launch at the startup. There are currently three listeners:</p>\n<ul>\n<li><code>hyprland</code> to listen Hyprland IPC events</li>\n<li><code>systemd_logind</code> to monitor for Systemd Logind events</li>\n<li><code>idle</code> which allows you to react when your computer has no activity</li>\n</ul>\n<p>Each one can take take on or more parameters. See reference below.</p>\n<p></p><h4>Configure Idling</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#configure-idling\"></a><p></p>\n<div><pre><code>def idle_config():\n return [\n [150, ['brightnessctl -s set 0'], ['brightnessctl -r']],\n [600, ['pidof hyprlock || hyprlock']],\n [720, ['hyprctl dispatch dpms off'], ['hyprctl dispatch dpms on']]\n ]\n</code></pre></div>\n<p>This method configure the Idle part. It should return the list of your idle actions. Each action has three parameters:</p>\n<ul>\n<li>The timeout in seconds</li>\n<li>The list of commands to execute when the timeout occurs</li>\n<li>The optional list of commands when the timeout is resumed (eg. once there is user activity after the timeout)</li>\n</ul>\n<p>Here, at 720 seconds, the screens are turned off. Then, if some activities are detected, they are turned on.</p>\n<p></p><h4>Configure monitors</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#configure-monitors\"></a><p></p>\n<div><pre><code>def on_hyprland_event(event, argument):\n if event in [ \"monitoradded\", \"monitorremoved\" ]:\n logger.info('Handling hyprland event: ' + event)\n set_monitors()\n</code></pre></div>\n<p>The method <code>on_hyprland_event</code> allows you to handle Hyprland IPC events. All events are avalable <a target=\"_blank\" href=\"https://wiki.hyprland.org/IPC/\">here</a>.</p>\n<p>In this case, when we connect or disconnect a monitor, we call a method to set our monitors according to our preferences. This method is as follows.</p>\n<div><pre><code>def set_monitors(self):\n logger.info('Setting monitors')\n if command.hyprland_get_monitor(description=\"HP 22es\") is not None:\n command.exec('hyprctl keyword monitor \"eDP-1,disable\"')\n else:\n command.exec('hyprctl keyword monitor \"eDP-1,preferred,auto,2\"')\n command.exec(\"brightnessctl -s set 0\")\n</code></pre></div>\n<p>We use the <code>hyprland_get_monitor</code> command helper to get the configuration of a particular monitor. <code>hyprland_get_monitor</code> accepts the name of the monitor, its description, the maker or the model. If the screen is not found, this method returns None.</p>\n<p>So, when we detect a \"HP 22es\" monitor, we disable the screen of the laptop. Otherwise, we turn on the monitor of the laptop, and we put the brightness at the lowest level (in my configuration, when I undock my laptop, the brightness is at 100%)</p>\n<p>For Sway, you have the <code>sway_get_monitor</code> helper method.</p>\n<p></p><h4>Configure suspend</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#configure-suspend\"></a><p></p>\n<div><pre><code>def on_PrepareForSleep(payload):\n if 'true' in payload:\n logger.info(\"Locking the screen before suspend\")\n command.exec(\"pidof hyprlock || hyprlock\", True)\n</code></pre></div>\n<p>The methods <code>on_Whatever(payload)</code> are automatically called when the signal <strong>Whatever</strong> is sent by Systemd Logind. Here, we are listening for the signal \"PrepareForSleep\" which is called just before you computer is suspending.</p>\n<p>The second parameter of the <code>command.exec</code> helper allows you to execute a command in the background. It's necessary here, otherwhise Easyland will wait indefinitely until the screen is unlocked.</p>\n<p></p><h4>Other tricks</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#other-tricks\"></a><p></p>\n<p>You may be interested to listen for Lock and Unlock events emitted from Systemd when you call <code>loginctl lock-session</code> and <code>loginctl unlock-session</code>. However, keep in mind that <code>hyprlock</code> and <code>swaylock</code> do not send any signal for these events, so you need to hack that.</p>\n<div><pre><code># To use this handler, you need to launch your locker (hyprlock or swaylock) like this: hyprlock &amp;&amp; loginctl unlock-session\ndef on_Unlock():\n logger.info(\"Unlocking the screen\")\n</code></pre></div>\n<p>To receive the Systemd <code>Unlock</code> signal, you should launch your screen locker with the following command: <code>hyprlock &amp;&amp; loginctl unlock-session</code>, so Systemd will send the <code>Unlock</code> signal when the screen is unlocked.</p>\n<p>For locking, keep in mind that <code>hyprlock</code> and <code>swaylock</code> do not listen for the Systemd <code>Lock</code> event, so you need to it manually.</p>\n<div><pre><code># To use this handler, you need to launch your locker like this: loginctl lock-session\ndef on_Lock():\n logger.info(\"Locking the screen\")\n command.exec('pidof hyprlock || hyprlock', True)\n # Do other actions if needed\n</code></pre></div>\n<p></p><h4>Alternative to on_WhateverSignal method</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#alternative-to-on_whateversignal-method\"></a><p></p>\n<p>Alternatively to write several methods to listen for Systemd events, you can also define method <code>on_systemd_event</code> and add a condition to achieve what you want:</p>\n<div><pre><code>def on_systemd_event(sender, signal, payload)\n if signal == 'Lock':\n ...\n if signal == 'PrepareForSleep':\n ...\n</code></pre></div>\n<p></p><h2>References</h2><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#references\"></a><p></p>\n<p></p><h3>Listeners parameters</h3><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#listeners-parameters\"></a><p></p>\n<p></p><h4>Hyprland listener parameters</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#hyprland-listener-parameters\"></a><p></p>\n<ul>\n<li><code>socket_path</code>: The path of the Hyprland IPC socket</li>\n</ul>\n<p></p><h4>Sway listener parameters</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#sway-listener-parameters\"></a><p></p>\n<ul>\n<li><code>event_types</code>: The type of events to be listened for</li>\n</ul>\n<p></p><h4>Idle listener parameters</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#idle-listener-parameters\"></a><p></p>\n<p>None.</p>\n<p></p><h4>systemd_logind parameters</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#systemd_logind-parameters\"></a><p></p>\n<p>None.</p>\n<p></p><h3>Listeners handler methods</h3><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#listeners-handler-methods\"></a><p></p>\n<table>\n<thead>\n<tr>\n<th>Sender</th>\n<th>Handler method to add to your class</th>\n<th>Arguments</th>\n</tr>\n</thead>\n<tbody>\n<tr>\n<td>Hyprland</td>\n<td>on_hyprland_event</td>\n<td>event, argument</td>\n</tr>\n<tr>\n<td>Sway</td>\n<td>on_sway_event_[type]</td>\n<td>payload</td>\n</tr>\n<tr>\n<td>Systemd Logind</td>\n<td>on_systemd_event</td>\n<td>sender, signal, payload</td>\n</tr>\n<tr>\n<td>Systemd Logind</td>\n<td>on_[signal]</td>\n<td>payload</td>\n</tr>\n</tbody>\n</table>\n<p></p><h4>Hyprland events</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#hyprland-events\"></a><p></p>\n<p>They are well documented <a target=\"_blank\" href=\"https://wiki.hyprland.org/IPC/\">here</a>.</p>\n<p></p><h4>Sway event types</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#sway-event-types\"></a><p></p>\n<p>For Sway, the current event types are those defined in the <a target=\"_blank\" href=\"https://man.archlinux.org/man/sway-ipc.7.en\">IPC manual</a></p>\n<p></p><h4>Systemd Logind events</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#systemd-logind-events\"></a><p></p>\n<p>These events are called \"signals\" in the Systemd terminology.</p>\n<p>They are not well documented but you can try to <a target=\"_blank\" href=\"https://www.freedesktop.org/software/systemd/man/latest/org.freedesktop.login1.html\">read that</a> (good luck).</p>\n<p>Some examples that can be useful:</p>\n<table>\n<thead>\n<tr>\n<th>Member</th>\n<th>Description</th>\n</tr>\n</thead>\n<tbody>\n<tr>\n<td>PrepareForShutdown</td>\n<td>Sent before a shutdown</td>\n</tr>\n<tr>\n<td>PrepareForSleep</td>\n<td>Sent before suspend</td>\n</tr>\n<tr>\n<td>Lock</td>\n<td>Sent when a lock is requested, eg <code>loginctrl lock-session</code></td>\n</tr>\n<tr>\n<td>Unlock</td>\n<td>Sent when an unlock is requested</td>\n</tr>\n<tr>\n<td>SessionNew</td>\n<td>When a session is created</td>\n</tr>\n</tbody>\n</table>\n<p>Keep in mind that these signals are independent from Wayland/Hyprland/Sway. My recommendation would be to always configure your compositor to use loginctl to send the signals, and add a listener in <strong>Easyland</strong> to achieve what you want.</p>\n<p></p><h4>Available helpers</h4><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#available-helpers\"></a><p></p>\n<table>\n<thead>\n<tr>\n<th>Method</th>\n<th>Usage</th>\n<th>Arguments</th>\n</tr>\n</thead>\n<tbody>\n<tr>\n<td>command.exec</td>\n<td>Execute a command, eventually in the background</td>\n<td>cmd (string), background (bool, default False), decode_json (bool, default False)</td>\n</tr>\n<tr>\n<td>command.hyprland_get_all_monitors</td>\n<td>Get all monitors and their configuration through Hyprland IPC</td>\n<td>None</td>\n</tr>\n<tr>\n<td>command.hyprland_get_monitor</td>\n<td>Get the config of one monitor, None if not found</td>\n<td>name, description, make, model</td>\n</tr>\n<tr>\n<td>command.sway_get_all_monitors</td>\n<td>Get all monitors and their configuration through Hyprland IPC</td>\n<td>None</td>\n</tr>\n<tr>\n<td>command.sway_get_monitor</td>\n<td>Get the config of one monitor, None if not found</td>\n<td>name, make, model</td>\n</tr>\n<tr>\n<td>logger</td>\n<td>Log messages to easyland.log and to STDOUT</td>\n<td>use logger.info, logger.error, for the severity etc.</td>\n</tr>\n<tr>\n<td>idle_config</td>\n<td>Set the idle configuration</td>\n<td>None</td>\n</tr>\n</tbody>\n</table>\n<p></p><h2>Contributions</h2><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#contributions\"></a><p></p>\n<ul>\n<li>Integrating other DBUS services should be easy with Easyland (type <code>dbusctl</code> to list all avalable DBUS on your system). Do not hesitate to let me know what you need.</li>\n<li>Better tests for Sway. I use Hyprland so feel free to submit bugs if you are using Sway and see an issue.</li>\n</ul>\n<p>If you see some bugs or propose patches, feel free to contribute.</p>\n<p></p><h2>Thanks</h2><a target=\"_blank\" href=\"https://github.com/juienpro/easyland#thanks\"></a><p></p>\n<p>Thanks to the developer(s) of <a target=\"_blank\" href=\"https://hyprland.org/\">Hyprland</a> for their fantastic compositor. I tried so many ones in the past, and this has been Hyprland that convinced me to do the switch from KDE :-)</p>\n</article></div>",
"author": "",
"favicon": "https://github.githubassets.com/favicons/favicon.svg",
"source": "github.com",
"published": "",
"ttr": 310,
"type": "object"
}