A plugin for the Hyprland compositor, implementing virtual-desktop functionality.
Go to file
2024-01-12 15:32:59 +01:00
.github first commit 2023-05-07 15:51:31 +02:00
include Update globals.hpp 2024-01-12 15:32:59 +01:00
src Update VirtualDeskManager.cpp 2024-01-12 15:30:18 +01:00
.clang-format first commit 2023-05-07 15:51:31 +02:00
.gitignore first commit 2023-05-07 15:51:31 +02:00
hyprload.toml chore: bump version to 2.0.1 2023-12-01 08:43:24 +01:00
LICENSE first commit 2023-05-07 15:51:31 +02:00
Makefile wip: support for new hyprpm 2023-12-10 10:49:02 +01:00
README.md chore: fix README 'thanks' section indentation 2023-11-19 17:12:51 +01:00

Virtual desktops for Hyprland hyprico

virtual-desktops is a plugin for the Hyprland compositor. virtual-desktops manages multiple screens workspaces as if they were a single virtual desktop.

Table of contents

What is this exactly?

In Hyprland, each screen has its own set of workspaces. For instance, say you have two monitors, with workspace 1 on screen 1 and workspace 2 on screen 2:

  • When you switch from workspace 1 to 2, Hyprland will simply focus your second screen;
  • If you switch to workspace 3, your active screen will go to workspace 3, whereas the other screen will stay on whichever workspace it is currently on.

You may think of a virtual desktop, instead, as a "single" workspace which extends across your screens (even though, internally, you will still have n different workspaces on your n monitors). If you've ever used KDE Plasma (or Gnome, I think) with multiple screens, this plugin basically replicates that functionality.
Taking the previous example:

  • You will be on virtual desktop 1. Let's say you open your web browser on your first screen and an IDE on your second screen;
  • When you switch to virtual desktop 2, both screens will switch to empty workspaces. Let's say here you open your email client and your favourite chat application;
  • If you switch back to virtual desktop 1, you will get back your web browser and the IDE on screen 1 and 2; and viceversa when you go back to virtual desktop 2.

WARNING: the main branch is currently on virtual-desktops v. 2.0b, which is beta software. If you're experiencing any bug, please open an issue here and then consider switching to v. 1.0

How does this work?

It's just workspaces, really

Internally, this simply ties n workspaces to your n screens, for each virtual desktop. That is, on virtual desktop 1 you will have workspace 1 on screen 1 and workspace 2 on screen 2; on virtual desktop 2, you will have workspace 3 on screen 1 and workspace 4 on screen 2, and so on.

However, if you focus another workspace on a given virtual desktop, the plugin will remember this and you will keep this layout (see Layouts).

Notice: screen 1 and screen 2 are not necessarily what you expect your first and second screen to be, e.g., screen 1 is not necessarily your left screen, and screen 2 is not necessarily your right screen.

Hyprctl dispatchers

This plugin exposes a few hyprctl dispatchers:

Dispatcher descritpion type example
vdesk [vdesk] Changes to virtual desktop vdesk see below vdesk 3 or vdesk coding
prevdesk Changes to previous virtual desktop none prevdesk
printdesk (vdesk) Prints to Hyprland log the specified vdesk or the currently active vdesk* (if no argument is given) optional vdesk, see below printdesk or printdesk 2 or printdesk coding
movetodesk vdesk(, window) Moves the active/selected window to the specified vdesk vdesk, optional window, see below movetodesk 2 or movetodesk 2,title:kitty
movetodesksilent vdesk(, window) same as movetodesk, but doesn't switch to desk same as above same as above
vdeskreset (vdesk) reset layouts on vdesk or on all vdesks if no argument is given (see Layouts) optional vdesk, see below vdeskreset or vdeskreset 2 or vdeskreset coding
nextdesk go to next vdesk. Creates it if it doesn't exist none nextdesk
cyclevdesks cycle between currently existing vdesks. Goes back to vdesk 1 if next vdesk does not exist none cyclevdesks
printlayout print to Hyprland logs the current layout none printlayout

*printdesk currently prints to the active Hyprland session log, thus probably not really useful.

For vdesk names, you can use:

  • ID: e.g., 1, 2 etc;
  • Name: e.g., coding, internet, mail and chats

If a vdesk with a given ID or name does not exist, it'll be created on the fly. If you give a (non configured, see below) name, it will be assigned to the next available vdesk id: the virtual-desktops plugin will remember this association even if Hyprland kills the related workspaces.

The movetodesk and movetodesksilent dispatchers work similarly to Hyprland's movetoworkspace and movetoworkspacesilent dispatchers. See Hyprland's wiki. Of course, make sure to use the vdesk syntax above instead of Hyprland's.

Mix with Hyprland native workspaces

You can use hyprctl dispatch vdesk n, even if you have no secondary screen connected at the moment (the behaviour would be identical to native workspaces). Also, I would REMOVE any workspace related configuration, such as wsbind. If you want to leverage workspace-specific rules, you can: workspaces are always assigned to the same vdesk given the same number of monitors, unless you focus (e.g. with hyprctl) another workspace (see Layouts). For instance:

  • Given two monitors:
    • vdesk 1 has workspaces 1 and 2;
    • vdesk 2 has workspaces 3 and 4, and so on;
  • Given three monitors:
    • vdesk 1 has workspaces 1, 2 and 3;
    • vdesk 2 has workspaces 4, 5 and 6, and so on.
  • Given four monitors...

The vdesk a workspace will end up to is easily computed by doing ceil(workspace_id / n_monitors). You know where I'm going with this one...you can easily script it.

Configuration values

This plugin exposes a few configuration options, under the plugin:virtual-desktops: category, namely:

Name description type example
names map a vdesk id with a name map[int:string], see below names = 1:coding, 2:internet, 3:mail and chats
cycleworkspaces if set to 1 and switching to the currently active vdesk, workspaces will be swapped between your monitors (see swapactiveworkspaces) 0 or 1 cycleworkspaces = 1
rememberlayout chooses how layouts should be remembered (see Layouts), defaults to size none, size or monitors remember = size
notifyinit chooses whether to display the startup notification, defaults to 1 0 or 1 notifyinit = 0
verbose_logging whether to log more stuff, defaults to 0 0 or 1 verbose_logging = 0
  • The names config option maps virtual desktop IDs to a name (you can then use this with the hyprctl dispatchers);
  • cycleworkspaces: THIS CURRENTLY DOES NOT WORK WITH MORE THAN 2 MONITORS. If you need this feature, please feel welcome to submit a PR ^^.

Example config

plugin {
    virtual-desktops {
        names = 1:coding, 2:internet, 3:mail and chats 
        cycleworkspaces = 1
        rememberlayout = size
        notifyinit = 0
        verbose_logging = 0
    }
}

Layouts

Version 2.0 of this plugin introduced the concept of a layout, with the meaning of "a specific combination of workspaces on a (more or less) specific combination of monitors".
In other words, virtual-desktops remembers if you focused another workspace on your vdesk, even if you switch to another vdesk and then come back to this one

Example

Say you have 2 monitors A and B, and you're on vdesk 1:

  • On monitor A you have workspace 1, and on monitor B you have workspace 2;
  • Now, say you focus workspace 4 with hyprctl dispatch workspace 4 on monitor B.
  • If you switch to vdesk 2 and back to vdesk 1, you will see workspace 4 on monitor B instead of workspace 2.

Notice that, in this case, workspace 4 would also be shown on vdesk 2.

Layouts are cached and restored if you disconnect/reconnect monitors

Internally, every vdesk will cache all the previous layouts. Once a monitor is connected/disconnected, the plugin will try to look for a previously existing layout for this configuration and it will apply it.

Example

Continuing from the previous example, say you now disconnect monitor B and then you reconnect it, while being on vdesk 1:

  • the previous layout will be matched and you will get back workspace 1 on monitor A and workspace 4 on monitor B.

This would work also if instead of disconnecting and reconnecting monitor B, you connected a third monitor C and then disconnect it.

Choosing how to remember, or choosing to forget

Version 2.0 also introduced the rememberlayout config option: with this option, we can choose if we want to match a layout by number of monitors (size) or by unique monitor descriptions (monitors). There is also the third and final option to not remember layouts at all (none).

Example

Continuing from the previous example. Say we now disconnect monitor B and connect monitor C: our connected monitors are A and C.

  • if rememberlayout = size (the default), the existing layout will still be matched, we will have workspace 1 on monitor A, workspace 4 on monitor C;
  • if rememberlayout = monitors, a new layout will be created with defaults: workspace 1 on monitor A, workspace 2 on monitor C;
  • if rememberlayout = none, same as above.

If we now disconnect monitor C and reconnect monitor B: our connected monitors are A and B.

  • if rememberlayout = size (the default), the existing layout will be matched, we will have workspace 1 on monitor A, workspace 4 on monitor B;
  • if rememberlayout = monitors, the existing layout will be matched, we will have workspace 1 on monitor A, workspace 4 on monitor B;
  • if rememberlayout = none, a new layout will be created with defaults: workspace 1 on monitor A, workspace 2 on monitor C.

Install

In order to use plugins, you should compile Hyprland yourself. See Hyprland Wiki#Using Plugins.

You can use:

HYPRLAND_HEADERS=path/to/hyprlandrepo make all

this will compile and copy the compiled .so plugin in the $HOME/.local/share/hyprload/plugins/bin path (useful if you use hyprload).
You can also use make virtual-desktops.so to output the compiled plugin in the repo directory.

Once compiled, you can tell Hyprland to load the plugin as described in the Hyprland wiki.

Help, Hyprland is being weird!

I've noticed that, sometimes, when disconnecting or reconnecting monitors, there might be weird artifacts or similar. Try running:
hyprctl reload

It's actually the plugin 😱

If instead you're seeing weird behaviour with the plugin itself, remember you can always run:
hyprtl dispatch vdeskreset

Thanks to

split-workspaces, from which I borrowed the Makefile, and the general idea of how to write Hyprland plugins.