Merge new river-classic
This commit is contained in:
@ -23,7 +23,7 @@ packages:
|
|||||||
- wget
|
- wget
|
||||||
- xz
|
- xz
|
||||||
sources:
|
sources:
|
||||||
- https://codeberg.org/river/river
|
- https://codeberg.org/river/river-classic
|
||||||
- https://gitlab.freedesktop.org/wlroots/wlroots.git
|
- https://gitlab.freedesktop.org/wlroots/wlroots.git
|
||||||
tasks:
|
tasks:
|
||||||
- install_deps: |
|
- install_deps: |
|
||||||
@ -42,13 +42,13 @@ tasks:
|
|||||||
sudo mv zig-x86_64-linux-0.15.1/zig /usr/bin/
|
sudo mv zig-x86_64-linux-0.15.1/zig /usr/bin/
|
||||||
sudo mv zig-x86_64-linux-0.15.1/lib /usr/lib/zig
|
sudo mv zig-x86_64-linux-0.15.1/lib /usr/lib/zig
|
||||||
- build: |
|
- build: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig build --summary all
|
zig build --summary all
|
||||||
- build_xwayland: |
|
- build_xwayland: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig build --summary all -Dxwayland
|
zig build --summary all -Dxwayland
|
||||||
- fmt: |
|
- fmt: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig fmt --check river/
|
zig fmt --check river/
|
||||||
zig fmt --check riverctl/
|
zig fmt --check riverctl/
|
||||||
zig fmt --check rivertile/
|
zig fmt --check rivertile/
|
||||||
|
|||||||
@ -20,7 +20,7 @@ packages:
|
|||||||
- wget
|
- wget
|
||||||
- xz
|
- xz
|
||||||
sources:
|
sources:
|
||||||
- https://codeberg.org/river/river
|
- https://codeberg.org/river/river-classic
|
||||||
- https://gitlab.freedesktop.org/wlroots/wlroots.git
|
- https://gitlab.freedesktop.org/wlroots/wlroots.git
|
||||||
tasks:
|
tasks:
|
||||||
- install_deps: |
|
- install_deps: |
|
||||||
@ -39,13 +39,13 @@ tasks:
|
|||||||
sudo mv zig-x86_64-linux-0.15.1/zig /usr/bin/
|
sudo mv zig-x86_64-linux-0.15.1/zig /usr/bin/
|
||||||
sudo mv zig-x86_64-linux-0.15.1/lib /usr/lib/zig
|
sudo mv zig-x86_64-linux-0.15.1/lib /usr/lib/zig
|
||||||
- build: |
|
- build: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig build --summary all
|
zig build --summary all
|
||||||
- build_xwayland: |
|
- build_xwayland: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig build --summary all -Dxwayland
|
zig build --summary all -Dxwayland
|
||||||
- fmt: |
|
- fmt: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig fmt --check river/
|
zig fmt --check river/
|
||||||
zig fmt --check riverctl/
|
zig fmt --check riverctl/
|
||||||
zig fmt --check rivertile/
|
zig fmt --check rivertile/
|
||||||
|
|||||||
@ -26,7 +26,7 @@ packages:
|
|||||||
- scdoc
|
- scdoc
|
||||||
- wget
|
- wget
|
||||||
sources:
|
sources:
|
||||||
- https://codeberg.org/river/river
|
- https://codeberg.org/river/river-classic
|
||||||
- https://gitlab.freedesktop.org/wlroots/wlroots.git
|
- https://gitlab.freedesktop.org/wlroots/wlroots.git
|
||||||
tasks:
|
tasks:
|
||||||
- install_deps: |
|
- install_deps: |
|
||||||
@ -46,13 +46,13 @@ tasks:
|
|||||||
sudo mv zig-x86_64-freebsd-0.15.1/zig /usr/bin/
|
sudo mv zig-x86_64-freebsd-0.15.1/zig /usr/bin/
|
||||||
sudo mv zig-x86_64-freebsd-0.15.1/lib /usr/lib/zig
|
sudo mv zig-x86_64-freebsd-0.15.1/lib /usr/lib/zig
|
||||||
- build: |
|
- build: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig build --summary all
|
zig build --summary all
|
||||||
- build_xwayland: |
|
- build_xwayland: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig build --summary all -Dxwayland
|
zig build --summary all -Dxwayland
|
||||||
- fmt: |
|
- fmt: |
|
||||||
cd river
|
cd river-classic
|
||||||
zig fmt --check river/
|
zig fmt --check river/
|
||||||
zig fmt --check riverctl/
|
zig fmt --check riverctl/
|
||||||
zig fmt --check rivertile/
|
zig fmt --check rivertile/
|
||||||
|
|||||||
50
PACKAGING.md
50
PACKAGING.md
@ -1,39 +1,39 @@
|
|||||||
# Packaging river for distribution
|
# Packaging river-classic for distribution
|
||||||
|
|
||||||
First of all, I apologize for writing river in Zig. It will likely make
|
First of all, I apologize for writing river-classic in Zig. It will likely make
|
||||||
your job harder until Zig is more mature/stable. I do however believe that
|
your job harder until Zig is more mature/stable. I do however believe that
|
||||||
writing my software in Zig allows me to deliver the best quality I can
|
writing my software in Zig allows me to deliver the best quality I can despite
|
||||||
despite the drawbacks of depending on a relatively immature language/toolchain.
|
the drawbacks of depending on a relatively immature language/toolchain.
|
||||||
|
|
||||||
## Source tarballs
|
## Source tarballs
|
||||||
|
|
||||||
Source tarballs with stable checksums and git submodule sources included may
|
Source tarballs with stable checksums may be found on the
|
||||||
be found on the [codeberg releases page](https://codeberg.org/river/river/releases).
|
[codeberg releases page](https://codeberg.org/river/river-classic/releases).
|
||||||
These tarballs are signed with the PGP key available on my website at
|
These tarballs are signed with the PGP key available on my website at
|
||||||
<https://isaacfreund.com/public_key.txt>.
|
<https://isaacfreund.com/public_key.txt>.
|
||||||
|
|
||||||
For the 0.1.3 release for example, the tarball and signature URLs are:
|
For the 0.3.12 release for example, the tarball and signature URLs are:
|
||||||
```
|
```
|
||||||
https://codeberg.org/river/river/releases/download/v0.1.3/river-0.1.3.tar.gz
|
https://codeberg.org/river/river-classic/releases/download/v0.1.3/river-classic-0.3.12.tar.gz
|
||||||
https://codeberg.org/river/river/releases/download/v0.1.3/river-0.1.3.tar.gz.sig
|
https://codeberg.org/river/river-classic/releases/download/v0.1.3/river-classic-0.3.12.tar.gz.sig
|
||||||
```
|
```
|
||||||
|
|
||||||
## Zig version
|
## Zig version
|
||||||
|
|
||||||
Until Zig 1.0, Zig releases will often have breaking changes that prevent
|
Until Zig 1.0, Zig releases will often have breaking changes that prevent
|
||||||
river from building. River tracks the latest minor version Zig release
|
river-classic from building. river-classic tracks the latest minor version Zig
|
||||||
and is only compatible with that release and any patch releases. At the time
|
release and is only compatible with that release and any patch releases. At the
|
||||||
of writing for example river is compatible with Zig 0.9.0 and 0.9.1 but
|
time of writing for example river is compatible with Zig 0.9.0 and 0.9.1 but not
|
||||||
not Zig 0.8.0 or 0.10.0.
|
Zig 0.8.0 or 0.10.0.
|
||||||
|
|
||||||
## Zig Package Manager
|
## Zig Package Manager
|
||||||
|
|
||||||
River uses the built-in Zig package manager for its (few) Zig dependencies.
|
river-classic uses the built-in Zig package manager for its (few) Zig
|
||||||
By default, running `zig build` will fetch river's Zig dependencies from the
|
dependencies. By default, running `zig build` will fetch river-classic's Zig
|
||||||
internet and store them in the global zig cache before building river. Since
|
dependencies from the internet and store them in the global zig cache before
|
||||||
accessing the internet is forbidden or at least frowned upon by most distro
|
building river-classic. Since accessing the internet is forbidden or at least
|
||||||
packaging infrastructure, there are ways to fetch the Zig dependencies in a
|
frowned upon by most distro packaging infrastructure, there are ways to fetch
|
||||||
separate step before building river:
|
the Zig dependencies in a separate step before building river-classic:
|
||||||
|
|
||||||
1. Fetch step with internet access:
|
1. Fetch step with internet access:
|
||||||
|
|
||||||
@ -114,10 +114,10 @@ in the first place. For greatest effect, both may be used.
|
|||||||
- `-Doptimize=ReleaseSmall`: Optimize for binary size,
|
- `-Doptimize=ReleaseSmall`: Optimize for binary size,
|
||||||
disable all assertions and runtime safety checks.
|
disable all assertions and runtime safety checks.
|
||||||
|
|
||||||
Please use `-Doptimize=ReleaseSafe` when building river for general
|
Please use `-Doptimize=ReleaseSafe` when building river-classic for general use.
|
||||||
use. CPU execution speed is not the performance bottleneck for river, the
|
CPU execution speed is not the performance bottleneck for river-classic, the GPU
|
||||||
GPU is. Additionally, the increased safety is more than worth the binary
|
is. Additionally, the increased safety is more than worth the binary size
|
||||||
size trade-off in my opinion.
|
trade-off in my opinion.
|
||||||
|
|
||||||
## Build prefix and DESTDIR
|
## Build prefix and DESTDIR
|
||||||
|
|
||||||
@ -126,7 +126,7 @@ environment variable. For example
|
|||||||
```bash
|
```bash
|
||||||
DESTDIR="/foo/bar" zig build --prefix /usr install
|
DESTDIR="/foo/bar" zig build --prefix /usr install
|
||||||
```
|
```
|
||||||
will install river to `/foo/bar/usr/bin/river`.
|
will install river-classic to `/foo/bar/usr/bin/river`.
|
||||||
|
|
||||||
The Zig build system only has a single install step, there is no way to build
|
The Zig build system only has a single install step, there is no way to build
|
||||||
artifacts for a given prefix and then install those artifacts to that prefix
|
artifacts for a given prefix and then install those artifacts to that prefix
|
||||||
@ -147,7 +147,7 @@ install() {
|
|||||||
## River specific suggestions
|
## River specific suggestions
|
||||||
|
|
||||||
I recommend installing the example init file found at `example/init` to
|
I recommend installing the example init file found at `example/init` to
|
||||||
`/usr/share/examples/river/init` or similar if your distribution has such
|
`/usr/share/examples/river-classic/init` or similar if your distribution has such
|
||||||
a convention.
|
a convention.
|
||||||
|
|
||||||
## Examples
|
## Examples
|
||||||
|
|||||||
69
README.md
69
README.md
@ -1,59 +1,52 @@
|
|||||||
<div align="center">
|
# river-classic
|
||||||
<img src="logo/logo_text_adaptive_color.svg" width="600em">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
## Overview
|
river-classic is a dynamic tiling Wayland compositor with flexible runtime
|
||||||
|
|
||||||
River is a dynamic tiling Wayland compositor with flexible runtime
|
|
||||||
configuration.
|
configuration.
|
||||||
|
|
||||||
Check [packaging status](https://repology.org/project/river-compositor/versions) —
|
It is a fork of [river](https://codeberg.org/river/river) 0.3 intended for users
|
||||||
|
that are happy with how river 0.3 works and do not wish to deal with the majorly
|
||||||
|
breaking changes planned for the river 0.4.0 release.
|
||||||
|
|
||||||
Join us at [#river](https://web.libera.chat/?channels=#river) on irc.libera.chat —
|
Join us at [#river](https://web.libera.chat/?channels=#river) on irc.libera.chat —
|
||||||
Read our man pages, [wiki](https://codeberg.org/river/wiki), and
|
Read our man pages, [wiki](https://codeberg.org/river/wiki), and
|
||||||
[Code of Conduct](CODE_OF_CONDUCT.md)
|
[Code of Conduct](CODE_OF_CONDUCT.md)
|
||||||
|
|
||||||
The main repository is on [codeberg](https://codeberg.org/river/river),
|
The main repository is on [codeberg](https://codeberg.org/river/river-classic),
|
||||||
which is where the issue tracker may be found and where contributions are accepted.
|
which is where the issue tracker may be found and where contributions are accepted.
|
||||||
|
|
||||||
Read-only mirrors exist on [sourcehut](https://git.sr.ht/~ifreund/river)
|
Read-only mirrors exist on [sourcehut](https://git.sr.ht/~ifreund/river-classic)
|
||||||
and [github](https://github.com/riverwm/river).
|
and [github](https://github.com/riverwm/river-classic).
|
||||||
|
|
||||||
*Note: river has not yet seen a stable 1.0 release and it will be necessary to
|
|
||||||
make significant breaking changes before 1.0 to realize my longer term plans.
|
|
||||||
That said, I do my best to avoid gratuitous breaking changes and bugs/crashes
|
|
||||||
should be rare. If you find a bug don't hesitate to
|
|
||||||
[open an issue](https://codeberg.org/river/river/issues/new/choose).*
|
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
Currently river's window management style is quite similar to
|
river-classic's window management style is quite similar to
|
||||||
[dwm](http://dwm.suckless.org), [xmonad](https://xmonad.org), and other classic
|
[dwm](http://dwm.suckless.org), [xmonad](https://xmonad.org), and other classic
|
||||||
dynamic tiling X11 window managers. Windows are automatically arranged in a tiled
|
dynamic tiling X11 window managers. Windows are automatically arranged in a tiled
|
||||||
layout and shifted around as windows are opened/closed.
|
layout and shifted around as windows are opened/closed.
|
||||||
|
|
||||||
Rather than having the tiled layout logic built into the compositor process,
|
Rather than having the tiled layout logic built into the compositor process,
|
||||||
river uses a [custom Wayland
|
river-classic uses a [custom Wayland
|
||||||
protocol](https://codeberg.org/river/river/src/branch/master/protocol/river-layout-v3.xml)
|
protocol](https://codeberg.org/river/river-classic/src/branch/master/protocol/river-layout-v3.xml)
|
||||||
and separate "layout generator" process. A basic layout generator, `rivertile`,
|
and separate "layout generator" process. A basic layout generator, `rivertile`,
|
||||||
is provided but users are encouraged to use community-developed [layout
|
is provided but users are encouraged to use community-developed [layout
|
||||||
generators](https://codeberg.org/river/wiki/src/branch/master/pages/Community-Layouts.md)
|
generators](https://codeberg.org/river/wiki/src/branch/master/pages/Community-Layouts.md)
|
||||||
or write their own. Examples in C and Python may be found
|
or write their own. Examples in C and Python may be found
|
||||||
[here](https://codeberg.org/river/river/src/branch/master/contrib).
|
[here](https://codeberg.org/river/river-classic/src/branch/master/contrib).
|
||||||
|
|
||||||
Tags are used to organize windows rather than workspaces. A window may be
|
Tags are used to organize windows rather than workspaces. A window may be
|
||||||
assigned to one or more tags. Likewise, one or more tags may be displayed on a
|
assigned to one or more tags. Likewise, one or more tags may be displayed on a
|
||||||
monitor at a time.
|
monitor at a time.
|
||||||
|
|
||||||
River is configured at runtime using the `riverctl` tool. It can define
|
river-classic is configured at runtime using the `riverctl` tool. It can define
|
||||||
keybindings, set the active layout generator, configure input devices, and more.
|
keybindings, set the active layout generator, configure input devices, and more.
|
||||||
On startup, river runs a user-defined init script which usually runs `riverctl`
|
On startup, river-classic runs a user-defined init script which usually runs
|
||||||
commands to set up the user's configuration.
|
`riverctl` commands to set up the user's configuration.
|
||||||
|
|
||||||
## Building
|
## Building
|
||||||
|
|
||||||
Note: If you are packaging river for distribution, see [PACKAGING.md](PACKAGING.md).
|
Note: If you are packaging river-classic for distribution, see [PACKAGING.md](PACKAGING.md).
|
||||||
|
|
||||||
To compile river first ensure that you have the following dependencies
|
To compile river-classic first ensure that you have the following dependencies
|
||||||
installed. The "development" versions are required if applicable to your
|
installed. The "development" versions are required if applicable to your
|
||||||
distribution.
|
distribution.
|
||||||
|
|
||||||
@ -79,7 +72,7 @@ Run `zig build -h` to see a list of all options.
|
|||||||
River can either be run nested in an X11/Wayland session or directly
|
River can either be run nested in an X11/Wayland session or directly
|
||||||
from a tty using KMS/DRM. Simply run the `river` command.
|
from a tty using KMS/DRM. Simply run the `river` command.
|
||||||
|
|
||||||
On startup river will run an executable file at `$XDG_CONFIG_HOME/river/init`
|
On startup river-classic will run an executable file at `$XDG_CONFIG_HOME/river/init`
|
||||||
if such an executable exists. If `$XDG_CONFIG_HOME` is not set,
|
if such an executable exists. If `$XDG_CONFIG_HOME` is not set,
|
||||||
`~/.config/river/init` will be used instead.
|
`~/.config/river/init` will be used instead.
|
||||||
|
|
||||||
@ -93,34 +86,14 @@ in the example directory.
|
|||||||
For complete documentation see the `river(1)`, `riverctl(1)`, and
|
For complete documentation see the `river(1)`, `riverctl(1)`, and
|
||||||
`rivertile(1)` man pages.
|
`rivertile(1)` man pages.
|
||||||
|
|
||||||
## Future Plans
|
|
||||||
|
|
||||||
Currently details such as how tags work across multiple monitors are not
|
|
||||||
possible for users to configure. It would be possible to extend river's source
|
|
||||||
code to allow more flexibility here but this comes at the cost of complexity and
|
|
||||||
there will always be someone who prefers something slightly different.
|
|
||||||
|
|
||||||
My long term plan to address this is to move as much window management policy as
|
|
||||||
possible out of the river compositor process and into the "layout generator"
|
|
||||||
process which will need to be renamed to "window manager." This will give users
|
|
||||||
much more power and control over river's behavior and also enable some really
|
|
||||||
cool workflows. For example, it would be possible to write a window manager in
|
|
||||||
lisp and use hot code reloading to edit its behavior it while it is running.
|
|
||||||
|
|
||||||
This is a non-trivial architectural change and will take a while to implement. I
|
|
||||||
plan to focus on this change for the 0.4.0 release cycle. Unfortunately, it will
|
|
||||||
almost certainly break existing river configurations as well. I think the
|
|
||||||
benefits outweigh that downside though and I will do my best to offer a
|
|
||||||
reasonable upgrade path.
|
|
||||||
|
|
||||||
## Donate
|
## Donate
|
||||||
|
|
||||||
If my work on river adds value to your life and you'd like to support me
|
If my work on river-classic adds value to your life and you'd like to support me
|
||||||
financially you can find donation information [here](https://isaacfreund.com/donate/).
|
financially you can find donation information [here](https://isaacfreund.com/donate/).
|
||||||
|
|
||||||
## Licensing
|
## Licensing
|
||||||
|
|
||||||
River is released under the GNU General Public License v3.0 only.
|
river-classic is released under the GNU General Public License v3.0 only.
|
||||||
|
|
||||||
The protocols in the `protocol` directory are released under various licenses by
|
The protocols in the `protocol` directory are released under various licenses by
|
||||||
various parties. You should refer to the copyright block of each protocol for
|
various parties. You should refer to the copyright block of each protocol for
|
||||||
|
|||||||
@ -1,11 +1,11 @@
|
|||||||
.{
|
.{
|
||||||
.name = .river,
|
.name = .river_classic,
|
||||||
// While a river release is in development, this string should contain
|
// While a river release is in development, this string should contain
|
||||||
// the version in development with the "-dev" suffix.
|
// the version in development with the "-dev" suffix.
|
||||||
// When a release is tagged, the "-dev" suffix should be removed for the
|
// When a release is tagged, the "-dev" suffix should be removed for the
|
||||||
// commit that gets tagged. Directly after the tagged commit, the version
|
// commit that gets tagged. Directly after the tagged commit, the version
|
||||||
// should be bumped and the "-dev" suffix added.
|
// should be bumped and the "-dev" suffix added.
|
||||||
.version = "0.3.12",
|
.version = "0.3.14-dev",
|
||||||
.paths = .{""},
|
.paths = .{""},
|
||||||
.dependencies = .{
|
.dependencies = .{
|
||||||
.pixman = .{
|
.pixman = .{
|
||||||
@ -25,5 +25,5 @@
|
|||||||
.hash = "xkbcommon-0.3.0-VDqIe3K9AQB2fG5ZeRcMC9i7kfrp5m2rWgLrmdNn9azr",
|
.hash = "xkbcommon-0.3.0-VDqIe3K9AQB2fG5ZeRcMC9i7kfrp5m2rWgLrmdNn9azr",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
.fingerprint = 0xf5e3672b8e8d6efc,
|
.fingerprint = 0x3dae7aba2ea52a3b,
|
||||||
}
|
}
|
||||||
|
|||||||
@ -36,6 +36,8 @@ wlr_layer_surface: *wlr.LayerSurfaceV1,
|
|||||||
scene_layer_surface: *wlr.SceneLayerSurfaceV1,
|
scene_layer_surface: *wlr.SceneLayerSurfaceV1,
|
||||||
popup_tree: *wlr.SceneTree,
|
popup_tree: *wlr.SceneTree,
|
||||||
|
|
||||||
|
newly_mapped: bool = false,
|
||||||
|
|
||||||
destroy: wl.Listener(*wlr.LayerSurfaceV1) = wl.Listener(*wlr.LayerSurfaceV1).init(handleDestroy),
|
destroy: wl.Listener(*wlr.LayerSurfaceV1) = wl.Listener(*wlr.LayerSurfaceV1).init(handleDestroy),
|
||||||
map: wl.Listener(void) = wl.Listener(void).init(handleMap),
|
map: wl.Listener(void) = wl.Listener(void).init(handleMap),
|
||||||
unmap: wl.Listener(void) = wl.Listener(void).init(handleUnmap),
|
unmap: wl.Listener(void) = wl.Listener(void).init(handleUnmap),
|
||||||
@ -96,19 +98,19 @@ fn handleDestroy(listener: *wl.Listener(*wlr.LayerSurfaceV1), _: *wlr.LayerSurfa
|
|||||||
|
|
||||||
fn handleMap(listener: *wl.Listener(void)) void {
|
fn handleMap(listener: *wl.Listener(void)) void {
|
||||||
const layer_surface: *LayerSurface = @fieldParentPtr("map", listener);
|
const layer_surface: *LayerSurface = @fieldParentPtr("map", listener);
|
||||||
const wlr_surface = layer_surface.wlr_layer_surface;
|
const wlr_layer_surface = layer_surface.wlr_layer_surface;
|
||||||
|
|
||||||
log.debug("layer surface '{s}' mapped", .{wlr_surface.namespace});
|
log.debug("layer surface '{s}' mapped", .{wlr_layer_surface.namespace});
|
||||||
|
|
||||||
layer_surface.output.arrangeLayers();
|
// This is a bit of a hack, but layer surfaces are not part of the
|
||||||
|
// transaction system in river 0.3 so there's not a significantly cleaner
|
||||||
const consider = wlr_surface.current.keyboard_interactive == .on_demand and
|
// way I see to do this.
|
||||||
(wlr_surface.current.layer == .top or wlr_surface.current.layer == .overlay);
|
layer_surface.newly_mapped = true;
|
||||||
handleKeyboardInteractiveExclusive(
|
|
||||||
layer_surface.output,
|
|
||||||
if (consider) layer_surface else null,
|
|
||||||
);
|
|
||||||
|
|
||||||
|
// Beware: it is possible for arrangeLayers() to destroy this LayerSurface!
|
||||||
|
const output = layer_surface.output;
|
||||||
|
output.arrangeLayers();
|
||||||
|
handleKeyboardInteractiveExclusive(output);
|
||||||
server.root.applyPending();
|
server.root.applyPending();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -117,8 +119,10 @@ fn handleUnmap(listener: *wl.Listener(void)) void {
|
|||||||
|
|
||||||
log.debug("layer surface '{s}' unmapped", .{layer_surface.wlr_layer_surface.namespace});
|
log.debug("layer surface '{s}' unmapped", .{layer_surface.wlr_layer_surface.namespace});
|
||||||
|
|
||||||
layer_surface.output.arrangeLayers();
|
// Beware: it is possible for arrangeLayers() to destroy this LayerSurface!
|
||||||
handleKeyboardInteractiveExclusive(layer_surface.output, null);
|
const output = layer_surface.output;
|
||||||
|
output.arrangeLayers();
|
||||||
|
handleKeyboardInteractiveExclusive(output);
|
||||||
server.root.applyPending();
|
server.root.applyPending();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,21 +141,25 @@ fn handleCommit(listener: *wl.Listener(*wlr.Surface), _: *wlr.Surface) void {
|
|||||||
if (wlr_layer_surface.initial_commit or
|
if (wlr_layer_surface.initial_commit or
|
||||||
@as(u32, @bitCast(wlr_layer_surface.current.committed)) != 0)
|
@as(u32, @bitCast(wlr_layer_surface.current.committed)) != 0)
|
||||||
{
|
{
|
||||||
layer_surface.output.arrangeLayers();
|
// Beware: it is possible for arrangeLayers() to destroy this LayerSurface!
|
||||||
handleKeyboardInteractiveExclusive(layer_surface.output, null);
|
const output = layer_surface.output;
|
||||||
|
output.arrangeLayers();
|
||||||
|
handleKeyboardInteractiveExclusive(output);
|
||||||
server.root.applyPending();
|
server.root.applyPending();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
layer_surface.newly_mapped = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Focus topmost keyboard-interactivity-exclusive layer surface above normal
|
fn handleKeyboardInteractiveExclusive(output: *Output) void {
|
||||||
/// content, or if none found, focus the surface given as `consider`.
|
|
||||||
/// Requires a call to Root.applyPending()
|
|
||||||
fn handleKeyboardInteractiveExclusive(output: *Output, consider: ?*LayerSurface) void {
|
|
||||||
if (server.lock_manager.state != .unlocked) return;
|
if (server.lock_manager.state != .unlocked) return;
|
||||||
|
|
||||||
// Find the topmost layer surface (if any) in the top or overlay layers which
|
// Find the topmost layer surface (if any) in the top or overlay layers which
|
||||||
// requests exclusive keyboard interactivity.
|
// requests exclusive keyboard interactivity.
|
||||||
const to_focus = outer: for ([_]zwlr.LayerShellV1.Layer{ .overlay, .top }) |layer| {
|
// If none is found, check for a newly mapped surface in the same layers with
|
||||||
|
// on demand keyboard interactivity.
|
||||||
|
const to_focus = blk: {
|
||||||
|
for ([_]zwlr.LayerShellV1.Layer{ .overlay, .top }) |layer| {
|
||||||
const tree = output.layerSurfaceTree(layer);
|
const tree = output.layerSurfaceTree(layer);
|
||||||
// Iterate in reverse to match rendering order.
|
// Iterate in reverse to match rendering order.
|
||||||
var it = tree.children.iterator(.reverse);
|
var it = tree.children.iterator(.reverse);
|
||||||
@ -163,11 +171,31 @@ fn handleKeyboardInteractiveExclusive(output: *Output, consider: ?*LayerSurface)
|
|||||||
if (wlr_layer_surface.surface.mapped and
|
if (wlr_layer_surface.surface.mapped and
|
||||||
wlr_layer_surface.current.keyboard_interactive == .exclusive)
|
wlr_layer_surface.current.keyboard_interactive == .exclusive)
|
||||||
{
|
{
|
||||||
break :outer layer_surface;
|
break :blk layer_surface;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else consider;
|
}
|
||||||
|
for ([_]zwlr.LayerShellV1.Layer{ .overlay, .top }) |layer| {
|
||||||
|
const tree = output.layerSurfaceTree(layer);
|
||||||
|
// Iterate in reverse to match rendering order.
|
||||||
|
var it = tree.children.iterator(.reverse);
|
||||||
|
while (it.next()) |node| {
|
||||||
|
assert(node.type == .tree);
|
||||||
|
if (@as(?*SceneNodeData, @ptrCast(@alignCast(node.data)))) |node_data| {
|
||||||
|
const layer_surface = node_data.data.layer_surface;
|
||||||
|
const wlr_layer_surface = layer_surface.wlr_layer_surface;
|
||||||
|
if (layer_surface.newly_mapped and
|
||||||
|
wlr_layer_surface.surface.mapped and
|
||||||
|
wlr_layer_surface.current.keyboard_interactive == .on_demand)
|
||||||
|
{
|
||||||
|
break :blk layer_surface;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break :blk null;
|
||||||
|
};
|
||||||
|
|
||||||
if (to_focus) |s| {
|
if (to_focus) |s| {
|
||||||
assert(s.wlr_layer_surface.current.keyboard_interactive != .none);
|
assert(s.wlr_layer_surface.current.keyboard_interactive != .none);
|
||||||
|
|||||||
@ -124,8 +124,6 @@ pub fn init(root: *Root) !void {
|
|||||||
const gamma_control_manager = try wlr.GammaControlManagerV1.create(server.wl_server);
|
const gamma_control_manager = try wlr.GammaControlManagerV1.create(server.wl_server);
|
||||||
scene.setGammaControlManagerV1(gamma_control_manager);
|
scene.setGammaControlManagerV1(gamma_control_manager);
|
||||||
|
|
||||||
if (server.linux_dmabuf) |linux_dmabuf| scene.setLinuxDmabufV1(linux_dmabuf);
|
|
||||||
|
|
||||||
const interactive_content = try scene.tree.createSceneTree();
|
const interactive_content = try scene.tree.createSceneTree();
|
||||||
const drag_icons = try scene.tree.createSceneTree();
|
const drag_icons = try scene.tree.createSceneTree();
|
||||||
const hidden_tree = try scene.tree.createSceneTree();
|
const hidden_tree = try scene.tree.createSceneTree();
|
||||||
|
|||||||
@ -78,12 +78,7 @@ fn getTarget(seat: *Seat, direction_str: []const u8, target_mode: TargetMode) !?
|
|||||||
if (seat.focused.view.pending.fullscreen) return null;
|
if (seat.focused.view.pending.fullscreen) return null;
|
||||||
if (target_mode == .skip_float and seat.focused.view.pending.float) return null;
|
if (target_mode == .skip_float and seat.focused.view.pending.float) return null;
|
||||||
const output = seat.focused_output orelse return null;
|
const output = seat.focused_output orelse return null;
|
||||||
|
if (seat.focused.view.pending.output != output) return null;
|
||||||
// If no currently view is focused, focus the first in the stack.
|
|
||||||
if (seat.focused != .view) {
|
|
||||||
var it = output.pending.wm_stack.iterator(.forward);
|
|
||||||
return it.next();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Logical direction, based on the view stack.
|
// Logical direction, based on the view stack.
|
||||||
if (std.meta.stringToEnum(Direction, direction_str)) |direction| {
|
if (std.meta.stringToEnum(Direction, direction_str)) |direction| {
|
||||||
|
|||||||
@ -188,8 +188,11 @@ pub fn logFn(
|
|||||||
|
|
||||||
const scope_prefix = if (scope == .default) ": " else "(" ++ @tagName(scope) ++ "): ";
|
const scope_prefix = if (scope == .default) ": " else "(" ++ @tagName(scope) ++ "): ";
|
||||||
|
|
||||||
var stderr = fs.File.stderr().writer(&.{});
|
var buffer: [256]u8 = undefined;
|
||||||
stderr.interface.print(level.asText() ++ scope_prefix ++ format ++ "\n", args) catch {};
|
const stderr = std.debug.lockStderrWriter(&buffer);
|
||||||
|
defer std.debug.unlockStderrWriter();
|
||||||
|
|
||||||
|
stderr.print(level.asText() ++ scope_prefix ++ format ++ "\n", args) catch {};
|
||||||
}
|
}
|
||||||
|
|
||||||
/// See wlroots_log_wrapper.c
|
/// See wlroots_log_wrapper.c
|
||||||
|
|||||||
Reference in New Issue
Block a user