i3 / sway like layout for hyprland.
Check the changelog for a list of new features and improvements
- i3 like tiling
- Node based window manipulation (you can interact with multiple windows at once)
- Greatly improved tabbed node groups over base hyprland
- Optional autotiling
Additional features may be suggested in the repo issues or the matrix room.
general.mp4
Important
The master branch of hy3 follows the master branch of hyprland. Attempting to use a mismatched hyprland release will result in failure when building or loading hy3.
To use hy3 against a release version of hyprland,
check out the matching hy3 tag for the hyprland version.
hy3 tags are formatted as hl{version}
where {version}
matches the release version of hyprland.
Hyprland now has a dedicated plugin manager, which should be used when your package manager isn't capable of locking hy3 builds to the correct hyprland version.
Important
Make sure hyprpm is activated by putting
exec-once = hyprpm reload -n
in your hyprland.conf. (See the wiki for details.)
To install hy3 via hyprpm run
hyprpm add https://github.com/outfoxxed/hy3
To update hy3 (and all other plugins), run
hyprpm update
(See the wiki for details.)
Warning
When you are running a tagged hyprland version hyprpm (0.34.0+) will build against hy3's
corrosponding release. However if you are running an untagged build (aka -git
) hyprpm
will build against hy3's latest commit. This means if you are running an out of date
untagged build of hyprland, hyprpm may pick an incompatible revision of hy3.
To fix this problem you will either need to update hyprland or manually build the correct version of hy3.
Assuming you use hyprland's home manager module, you can easily integrate hy3 by adding it to the plugins array.
# flake.nix
{
inputs = {
nixpkgs.url = "github:NixOS/nixpkgs/nixos-unstable";
home-manager = {
url = "github:nix-community/home-manager";
inputs.nixpkgs.follows = "nixpkgs";
};
hyprland.url = "git+https://github.com/hyprwm/Hyprland?submodules=1&ref={version}";
# where {version} is the hyprland release version
# or "github:hyprwm/Hyprland" to follow the development branch
hy3 = {
url = "github:outfoxxed/hy3?ref=hl{version}"; # where {version} is the hyprland release version
# or "github:outfoxxed/hy3" to follow the development branch.
# (you may encounter issues if you dont do the same for hyprland)
inputs.hyprland.follows = "hyprland";
};
};
outputs = { nixpkgs, home-manager, hyprland, hy3, ... }: {
homeConfigurations."user@hostname" = home-manager.lib.homeManagerConfiguration {
pkgs = nixpkgs.legacyPackages.x86_64-linux;
modules = [
hyprland.homeManagerModules.default
{
wayland.windowManager.hyprland = {
enable = true;
plugins = [ hy3.packages.x86_64-linux.hy3 ];
};
}
];
};
};
}
hy3's binary is availible as ${hy3.packages.<system>.hy3}/lib/libhy3.so
, so you can also
directly use it in your hyprland config like so:
# ...
wayland.windowManager.hyprland = {
# ...
extraConfig = ''
plugin = ${hy3.packages.x86_64-linux.hy3}/lib/libhy3.so
'';
};
Hyprland now has a dedicated plugin manager, which should be used when your package manager isn't capable of locking hy3 builds to the correct hyprland version.
Important
Make sure hyprpm is activated by putting
exec-once = hyprpm reload -n
in your hyprland.conf. (See the wiki for details.)
To install hy3 via hyprpm run
hyprpm add https://github.com/outfoxxed/hy3
To update hy3 (and all other plugins), run
hyprpm update
Sometimes the headers from hyprland are not updated, if this happens run (See issue #109 for an example of where this happened)
hyprpm update -f
(See the wiki for details.)
Warning
When you are running a tagged hyprland version hyprpm (0.34.0+) will build against hy3's
corrosponding release. However if you are running an untagged build (aka -git
) hyprpm
will build against hy3's latest commit. This means if you are running an out of date
untagged build of hyprland, hyprpm may pick an incompatible revision of hy3.
To fix this problem you will either need to update hyprland or manually build the correct version of hy3.
Install hyprland, including its headers and pkg-config file, then run the following commands:
cmake -DCMAKE_BUILD_TYPE=Release -B build
cmake --build build
The plugin will be located at build/libhy3.so
, and you can load it normally
(See the hyprland wiki for details.)
Note that the hyprland headers and pkg-config file MUST be installed correctly, for the target version of hyprland.
Important
The configuration listed below is for the current hy3 commit. If you are using a release version of hy3 then make sure you are reading the tagged revision of this readme.
Set your general:layout
to hy3
in hyprland.conf.
hy3 requires using a few custom dispatchers for normal operation. In your hyprland config replace the following dispatchers:
movefocus
->hy3:movefocus
movewindow
->hy3:movewindow
You can use hy3:makegroup
to create a new split.
The dispatcher list and config fields sections have all the configuration options, and some explanation as to what they do. The hyprland config in my dots can also be used as a reference.
plugin {
hy3 {
# disable gaps when only one window is onscreen
# 0 - always show gaps
# 1 - hide gaps with a single window onscreen
# 2 - 1 but also show the window border
no_gaps_when_only = <int> # default: 0
# policy controlling what happens when a node is removed from a group,
# leaving only a group
# 0 = remove the nested group
# 1 = keep the nested group
# 2 = keep the nested group only if its parent is a tab group
node_collapse_policy = <int> # default: 2
# policy controlling what windows will be focused using `hy3:movefocused`
# 0 = focus strictly by layout, don't attempt to skip windows that are obscured by another one
# 1 = do not focus windows which are entirely obscured by a floating window
# 2 = when `hy3:movefocus` layer is `samelayer` then use focus policy 0 (focus strictly by layout)
# when `hy3:movefocus` layer is `all` then use focus policy 1 (don't focus obscured windows)
focus_obscured_windows_policy = <int> # default: 2
# which layers should be considered when changing focus with `hy3:movefocus`?
# samelayer = only focus windows on same layer as the source window (floating or tiled)
# all = choose the closest window irrespective of the layout
# tiled = only choose tiled windows, not especially useful but permitted by parser
# floating = only choose floating windows, not especially useful but permitted by parser
default_movefocus_layer = <string> # default: `samelayer`
# offset from group split direction when only one window is in a group
group_inset = <int> # default: 10
# scale factor of windows on the special workspace
special_scale_factor = <float> # default: 0.8
# if a tab group will automatically be created for the first window spawned in a workspace
tab_first_window = <bool>
# tab group settings
tabs {
# height of the tab bar
height = <int> # default: 15
# padding between the tab bar and its focused node
padding = <int> # default: 5
# the tab bar should animate in/out from the top instead of below the window
from_top = <bool> # default: false
# rounding of tab bar corners
rounding = <int> # default: 3
# render the window title on the bar
render_text = <bool> # default: true
# center the window title
text_center = <bool> # default: false
# font to render the window title with
text_font = <string> # default: Sans
# height of the window title
text_height = <int> # default: 8
# left padding of the window title
text_padding = <int> # default: 3
# active tab bar segment color
col.active = <color> # default: 0xff32b4ff
# urgent tab bar segment color
col.urgent = <color> # default: 0xffff4f4f
# inactive tab bar segment color
col.inactive = <color> # default: 0x80808080
# active tab bar text color
col.text.active = <color> # default: 0xff000000
# urgent tab bar text color
col.text.urgent = <color> # default: 0xff000000
# inactive tab bar text color
col.text.inactive = <color> # default: 0xff000000
}
# autotiling settings
autotile {
# enable autotile
enable = <bool> # default: false
# make autotile-created groups ephemeral
ephemeral_groups = <bool> # default: true
# if a window would be squished smaller than this width, a vertical split will be created
# -1 = never automatically split vertically
# 0 = always automatically split vertically
# <number> = pixel height to split at
trigger_width = <int> # default: 0
# if a window would be squished smaller than this height, a horizontal split will be created
# -1 = never automatically split horizontally
# 0 = always automatically split horizontally
# <number> = pixel height to split at
trigger_height = <int> # default: 0
# a space or comma separated list of workspace ids where autotile should be enabled
# it's possible to create an exception rule by prefixing the definition with "not:"
# workspaces = 1,2 # autotiling will only be enabled on workspaces 1 and 2
# workspaces = not:1,2 # autotiling will be enabled on all workspaces except 1 and 2
workspaces = <string> # default: all
}
}
}
hy3:makegroup, <h | v | opposite | tab>, [ephemeral | force_ephemeral]
- make a vertical / horizontal split or tab groupephemeral
- the group will be removed once it contains only one node. does not affect existing groups.force_ephemeral
- same as ephemeral, but converts existing single windows groups.
hy3:changegroup, <h | v | tab | untab | toggletab | opposite>
- change the group the node belongs to, to a different layoutuntab
will untab the group if it was previously tabbedtoggletab
will untab if group is tabbed, and tab if group is untabbedopposite
will toggle between horizontal and vertical layouts if the group is not tabbed.
hy3:setephemeral, <true | false>
- change the ephemerality of the group the node belongs tohy3:movefocus, <l | u | d | r | left | down | up | right>, [visible], [warp | nowarp]
- move the focus left, up, down, or rightvisible
- only move between visible nodes, not hidden tabswarp
- warp the mouse to the selected window, even ifgeneral:no_cursor_warps
is true.nowarp
- does not warp the mouse to the selected window, even ifgeneral:no_cursor_warps
is false.
hy3:warpcursor
- warp the cursor to the center of the focused nodehy3:movewindow, <l | u | d | r | left | down | up | right>, [once], [visible]
- move a window left, up, down, or rightonce
- only move directly to the neighboring group, without moving into any of its subgroupsvisible
- only move between visible nodes, not hidden tabs
hy3:movetoworkspace, <workspace>, [follow]
- move the active node to the given workspacefollow
- change focus to the given workspace when moving the selected node
hy3:killactive
- close all windows in the focused nodehy3:changefocus, <top | bottom | raise | lower | tab | tabnode>
top
- focus all nodes in the workspacebottom
- focus the single root selection windowraise
- raise focus one levellower
- lower focus one leveltab
- raise focus to the nearest tabtabnode
- raise focus to the nearest node under the tab
hy3:focustab <mouse | [l | r | left | right | index, <index>], [prioritize_hovered | require_hovered], [wrap]>
mouse
- focus the tab under the mouse, works well with a non consuming bind, e.g.# binds hy3:focustab to lmb and still allows windows to receive clicks bindn = , mouse:272, hy3:focustab, mouse
l | r | left | right
- direction to change focus towardsindex, <index>
- select theindex
th tabprioritize_hovered
- prioritize the tab group under the mouse when multiple are stacked. use the lowest group if none is under the mouse.require_hovered
- affect the tab group under the mouse. do nothing if none are hovered.wrap
- wrap to the opposite size of the tab bar if moving off the end
hy3:debugnodes
- print the node tree into the hyprland loghy3:resizenode, <vector>
- like Hyprlandresizeactive
, but applied to the whole focused group instead of just a window⚠️ ALPHA QUALITYhy3:setswallow, <true | false | toggle>
- set the containing node's window swallow state⚠️ ALPHA QUALITYhy3:expand, <expand | shrink | base>
- expand the current node to cover other nodesexpand
- expand by one nodeshrink
- shrink by one nodebase
- undo all expansions