View: rework configure abstraction
- Move the decision whether a configure should be tracked or not into the xdg toplevel/xwayland code. - Only track configures for xdg toplevels with the transaction system if the dimensions of the view are affected.
This commit is contained in:
parent
915fb7ae7b
commit
b4ae62cd40
@ -497,18 +497,13 @@ fn sendConfigures(root: *Self) void {
|
|||||||
// This can happen if a view is unmapped while a layout demand including it is inflight
|
// This can happen if a view is unmapped while a layout demand including it is inflight
|
||||||
if (!view.mapped) continue;
|
if (!view.mapped) continue;
|
||||||
|
|
||||||
if (view.needsConfigure()) {
|
if (view.configure()) {
|
||||||
view.configure();
|
|
||||||
|
|
||||||
// We don't give a damn about frame perfection for xwayland views
|
|
||||||
if (!build_options.xwayland or view.impl != .xwayland_view) {
|
|
||||||
root.inflight_configures += 1;
|
root.inflight_configures += 1;
|
||||||
view.saveSurfaceTree();
|
view.saveSurfaceTree();
|
||||||
view.sendFrameDone();
|
view.sendFrameDone();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (root.inflight_configures > 0) {
|
if (root.inflight_configures > 0) {
|
||||||
std.log.scoped(.transaction).debug("started transaction with {} pending configure(s)", .{
|
std.log.scoped(.transaction).debug("started transaction with {} pending configure(s)", .{
|
||||||
@ -583,8 +578,6 @@ fn commitTransaction(root: *Self) void {
|
|||||||
while (focus_stack_it.next()) |view| {
|
while (focus_stack_it.next()) |view| {
|
||||||
assert(view.inflight.output == output);
|
assert(view.inflight.output == output);
|
||||||
|
|
||||||
view.inflight_serial = null;
|
|
||||||
|
|
||||||
if (view.current.output != view.inflight.output or
|
if (view.current.output != view.inflight.output or
|
||||||
(output.current.fullscreen == view and output.inflight.fullscreen != view))
|
(output.current.fullscreen == view and output.inflight.fullscreen != view))
|
||||||
{
|
{
|
||||||
|
@ -139,9 +139,6 @@ inflight_wm_stack_link: wl.list.Link,
|
|||||||
|
|
||||||
current: State = .{},
|
current: State = .{},
|
||||||
|
|
||||||
/// The serial sent with the currently inflight configure event
|
|
||||||
inflight_serial: ?u32 = null,
|
|
||||||
|
|
||||||
/// The floating dimensions the view, saved so that they can be restored if the
|
/// The floating dimensions the view, saved so that they can be restored if the
|
||||||
/// view returns to floating mode.
|
/// view returns to floating mode.
|
||||||
float_box: wlr.Box = undefined,
|
float_box: wlr.Box = undefined,
|
||||||
@ -224,6 +221,9 @@ pub fn updateCurrent(view: *Self) void {
|
|||||||
|
|
||||||
view.current = view.inflight;
|
view.current = view.inflight;
|
||||||
view.dropSavedSurfaceTree();
|
view.dropSavedSurfaceTree();
|
||||||
|
if (view.impl == .xdg_toplevel) {
|
||||||
|
view.impl.xdg_toplevel.configure_state = .idle;
|
||||||
|
}
|
||||||
|
|
||||||
const color = blk: {
|
const color = blk: {
|
||||||
if (view.current.urgent) break :blk &config.border_color_urgent;
|
if (view.current.urgent) break :blk &config.border_color_urgent;
|
||||||
@ -259,22 +259,16 @@ pub fn updateCurrent(view: *Self) void {
|
|||||||
view.borders.bottom.setColor(color);
|
view.borders.bottom.setColor(color);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn needsConfigure(self: Self) bool {
|
/// Returns true if the configure should be waited for by the transaction system.
|
||||||
assert(self.mapped);
|
pub fn configure(self: *Self) bool {
|
||||||
return switch (self.impl) {
|
|
||||||
.xdg_toplevel => |xdg_toplevel| xdg_toplevel.needsConfigure(),
|
|
||||||
.xwayland_view => |xwayland_view| xwayland_view.needsConfigure(),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn configure(self: *Self) void {
|
|
||||||
assert(self.mapped and !self.destroying);
|
assert(self.mapped and !self.destroying);
|
||||||
switch (self.impl) {
|
switch (self.impl) {
|
||||||
.xdg_toplevel => |*xdg_toplevel| xdg_toplevel.configure(),
|
.xdg_toplevel => |*xdg_toplevel| return xdg_toplevel.configure(),
|
||||||
.xwayland_view => |*xwayland_view| {
|
.xwayland_view => |*xwayland_view| {
|
||||||
// TODO(zig): remove this uneeded if statement
|
// TODO(zig): remove this uneeded if statement
|
||||||
// https://github.com/ziglang/zig/issues/13655
|
// https://github.com/ziglang/zig/issues/13655
|
||||||
if (build_options.xwayland) xwayland_view.configure();
|
if (build_options.xwayland) return xwayland_view.configure();
|
||||||
|
unreachable;
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
const std = @import("std");
|
const std = @import("std");
|
||||||
|
const assert = std.debug.assert;
|
||||||
const math = std.math;
|
const math = std.math;
|
||||||
const wlr = @import("wlroots");
|
const wlr = @import("wlroots");
|
||||||
const wl = @import("wayland").server.wl;
|
const wl = @import("wayland").server.wl;
|
||||||
@ -39,8 +40,14 @@ xdg_toplevel: *wlr.XdgToplevel,
|
|||||||
/// Initialized on map
|
/// Initialized on map
|
||||||
geometry: wlr.Box = undefined,
|
geometry: wlr.Box = undefined,
|
||||||
|
|
||||||
/// Set to true when the client acks the configure with serial View.inflight_serial.
|
configure_state: union(enum) {
|
||||||
acked_inflight_serial: bool = false,
|
/// No configure has been sent since the last configure was acked.
|
||||||
|
idle,
|
||||||
|
/// A configure was sent with the given serial but has not yet been acked.
|
||||||
|
inflight: u32,
|
||||||
|
/// A configure was acked but the surface has not yet been committed.
|
||||||
|
acked,
|
||||||
|
} = .idle,
|
||||||
|
|
||||||
// Listeners that are always active over the view's lifetime
|
// Listeners that are always active over the view's lifetime
|
||||||
destroy: wl.Listener(void) = wl.Listener(void).init(handleDestroy),
|
destroy: wl.Listener(void) = wl.Listener(void).init(handleDestroy),
|
||||||
@ -83,43 +90,52 @@ pub fn create(xdg_toplevel: *wlr.XdgToplevel) error{OutOfMemory}!void {
|
|||||||
_ = xdg_toplevel.setWmCapabilities(.{ .fullscreen = true });
|
_ = xdg_toplevel.setWmCapabilities(.{ .fullscreen = true });
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns true if a configure must be sent to ensure that the inflight
|
/// Send a configure event, applying the inflight state of the view.
|
||||||
/// dimensions are applied.
|
pub fn configure(self: *Self) bool {
|
||||||
pub fn needsConfigure(self: Self) bool {
|
assert(self.configure_state == .idle);
|
||||||
const view = self.view;
|
|
||||||
|
const inflight = &self.view.inflight;
|
||||||
|
const current = &self.view.current;
|
||||||
|
|
||||||
// We avoid a special case for newly mapped views which we have not yet
|
// We avoid a special case for newly mapped views which we have not yet
|
||||||
// configured by setting the current width/height to the initial width/height
|
// configured by setting the current width/height to the initial width/height
|
||||||
// of the view in handleMap().
|
// of the view in handleMap().
|
||||||
return view.inflight.box.width != view.current.box.width or
|
if (inflight.box.width == current.box.width and
|
||||||
view.inflight.box.height != view.current.box.height or
|
inflight.box.height == current.box.height and
|
||||||
(view.inflight.focus != 0) != (view.current.focus != 0) or
|
(inflight.focus != 0) == (current.focus != 0) and
|
||||||
(view.inflight.output != null and view.inflight.output.?.inflight.fullscreen == view) !=
|
(inflight.output != null and inflight.output.?.inflight.fullscreen == self.view) ==
|
||||||
(view.current.output != null and view.current.output.?.current.fullscreen == view) or
|
(current.output != null and current.output.?.current.fullscreen == self.view) and
|
||||||
view.inflight.borders != view.current.borders or
|
inflight.borders == current.borders and
|
||||||
view.inflight.resizing != view.current.resizing;
|
inflight.resizing == current.resizing)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Send a configure event, applying the inflight state of the view.
|
_ = self.xdg_toplevel.setActivated(inflight.focus != 0);
|
||||||
pub fn configure(self: *Self) void {
|
|
||||||
const state = &self.view.inflight;
|
|
||||||
|
|
||||||
self.view.inflight_serial = self.xdg_toplevel.setSize(state.box.width, state.box.height);
|
const fullscreen = inflight.output != null and inflight.output.?.inflight.fullscreen == self.view;
|
||||||
|
|
||||||
_ = self.xdg_toplevel.setActivated(state.focus != 0);
|
|
||||||
|
|
||||||
const fullscreen = state.output != null and state.output.?.inflight.fullscreen == self.view;
|
|
||||||
_ = self.xdg_toplevel.setFullscreen(fullscreen);
|
_ = self.xdg_toplevel.setFullscreen(fullscreen);
|
||||||
|
|
||||||
if (state.borders) {
|
if (inflight.borders) {
|
||||||
_ = self.xdg_toplevel.setTiled(.{ .top = true, .bottom = true, .left = true, .right = true });
|
_ = self.xdg_toplevel.setTiled(.{ .top = true, .bottom = true, .left = true, .right = true });
|
||||||
} else {
|
} else {
|
||||||
_ = self.xdg_toplevel.setTiled(.{ .top = false, .bottom = false, .left = false, .right = false });
|
_ = self.xdg_toplevel.setTiled(.{ .top = false, .bottom = false, .left = false, .right = false });
|
||||||
}
|
}
|
||||||
|
|
||||||
_ = self.xdg_toplevel.setResizing(state.resizing);
|
_ = self.xdg_toplevel.setResizing(inflight.resizing);
|
||||||
|
|
||||||
self.acked_inflight_serial = false;
|
// Only track configures with the transaction system if they affect the dimensions of the view.
|
||||||
|
if (inflight.box.width == current.box.width and
|
||||||
|
inflight.box.height == current.box.height)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.configure_state = .{
|
||||||
|
.inflight = self.xdg_toplevel.setSize(inflight.box.width, inflight.box.height),
|
||||||
|
};
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn rootSurface(self: Self) *wlr.Surface {
|
pub fn rootSurface(self: Self) *wlr.Surface {
|
||||||
@ -240,10 +256,11 @@ fn handleAckConfigure(
|
|||||||
acked_configure: *wlr.XdgSurface.Configure,
|
acked_configure: *wlr.XdgSurface.Configure,
|
||||||
) void {
|
) void {
|
||||||
const self = @fieldParentPtr(Self, "ack_configure", listener);
|
const self = @fieldParentPtr(Self, "ack_configure", listener);
|
||||||
if (self.view.inflight_serial) |serial| {
|
switch (self.configure_state) {
|
||||||
if (serial == acked_configure.serial) {
|
.inflight => |serial| if (acked_configure.serial == serial) {
|
||||||
self.acked_inflight_serial = true;
|
self.configure_state = .acked;
|
||||||
}
|
},
|
||||||
|
.acked, .idle => {},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -263,36 +280,36 @@ fn handleCommit(listener: *wl.Listener(*wlr.Surface), _: *wlr.Surface) void {
|
|||||||
|
|
||||||
const old_geometry = self.geometry;
|
const old_geometry = self.geometry;
|
||||||
self.xdg_toplevel.base.getGeometry(&self.geometry);
|
self.xdg_toplevel.base.getGeometry(&self.geometry);
|
||||||
|
|
||||||
|
switch (self.configure_state) {
|
||||||
|
.idle => {
|
||||||
const size_changed = self.geometry.width != old_geometry.width or
|
const size_changed = self.geometry.width != old_geometry.width or
|
||||||
self.geometry.height != old_geometry.height;
|
self.geometry.height != old_geometry.height;
|
||||||
|
const no_layout = view.current.output != null and view.current.output.?.layout == null;
|
||||||
|
|
||||||
if (view.inflight_serial != null) {
|
if (size_changed and (view.current.float or no_layout) and !view.current.fullscreen) {
|
||||||
if (self.acked_inflight_serial) {
|
|
||||||
view.inflight_serial = null;
|
|
||||||
server.root.notifyConfigured();
|
|
||||||
} else {
|
|
||||||
// If the client has not yet acked our configure, we need to send a
|
|
||||||
// frame done event so that it commits another buffer. These
|
|
||||||
// buffers won't be rendered since we are still rendering our
|
|
||||||
// stashed buffer from when the transaction started.
|
|
||||||
view.sendFrameDone();
|
|
||||||
}
|
|
||||||
} else if (size_changed and !view.current.fullscreen and
|
|
||||||
(view.current.float or view.current.output == null or view.current.output.?.layout == null))
|
|
||||||
{
|
|
||||||
log.info(
|
log.info(
|
||||||
"client initiated size change: {}x{} -> {}x{}",
|
"client initiated size change: {}x{} -> {}x{}",
|
||||||
.{ old_geometry.width, old_geometry.height, self.geometry.width, self.geometry.height },
|
.{ old_geometry.width, old_geometry.height, self.geometry.width, self.geometry.height },
|
||||||
);
|
);
|
||||||
|
|
||||||
// If the client has decided to resize itself and the view is floating,
|
|
||||||
// then respect that resize.
|
|
||||||
view.current.box.width = self.geometry.width;
|
view.current.box.width = self.geometry.width;
|
||||||
view.current.box.height = self.geometry.height;
|
view.current.box.height = self.geometry.height;
|
||||||
view.pending.box.width = self.geometry.width;
|
view.pending.box.width = self.geometry.width;
|
||||||
view.pending.box.height = self.geometry.height;
|
view.pending.box.height = self.geometry.height;
|
||||||
server.root.applyPending();
|
server.root.applyPending();
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
// If the client has not yet acked our configure, we need to send a
|
||||||
|
// frame done event so that it commits another buffer. These
|
||||||
|
// buffers won't be rendered since we are still rendering our
|
||||||
|
// stashed buffer from when the transaction started.
|
||||||
|
.inflight => view.sendFrameDone(),
|
||||||
|
.acked => {
|
||||||
|
self.configure_state = .idle;
|
||||||
|
server.root.notifyConfigured();
|
||||||
|
},
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Called when the client asks to be fullscreened. We always honor the request
|
/// Called when the client asks to be fullscreened. We always honor the request
|
||||||
|
@ -79,38 +79,39 @@ pub fn create(xwayland_surface: *wlr.XwaylandSurface) error{OutOfMemory}!void {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn needsConfigure(self: Self) bool {
|
/// Always returns false as we do not care about frame perfection for Xwayland views.
|
||||||
|
pub fn configure(self: Self) bool {
|
||||||
const output = self.view.inflight.output orelse return false;
|
const output = self.view.inflight.output orelse return false;
|
||||||
|
|
||||||
var output_box: wlr.Box = undefined;
|
var output_box: wlr.Box = undefined;
|
||||||
server.root.output_layout.getBox(output.wlr_output, &output_box);
|
server.root.output_layout.getBox(output.wlr_output, &output_box);
|
||||||
|
|
||||||
const view = self.view;
|
const inflight = &self.view.inflight;
|
||||||
return self.xwayland_surface.x != view.inflight.box.x + output_box.x or
|
const current = &self.view.current;
|
||||||
self.xwayland_surface.y != view.inflight.box.y + output_box.y or
|
|
||||||
self.xwayland_surface.width != view.inflight.box.width or
|
if (self.xwayland_surface.x == inflight.box.x + output_box.x and
|
||||||
self.xwayland_surface.height != view.inflight.box.height or
|
self.xwayland_surface.y == inflight.box.y + output_box.y and
|
||||||
(view.inflight.focus != 0) != (view.current.focus != 0) or
|
self.xwayland_surface.width == inflight.box.width and
|
||||||
(view.inflight.output != null and view.inflight.output.?.inflight.fullscreen == view) !=
|
self.xwayland_surface.height == inflight.box.height and
|
||||||
(view.current.output != null and view.current.output.?.current.fullscreen == view);
|
(inflight.focus != 0) == (current.focus != 0) and
|
||||||
|
(output.inflight.fullscreen == self.view) ==
|
||||||
|
(current.output != null and current.output.?.current.fullscreen == self.view))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn configure(self: Self) void {
|
|
||||||
const output = self.view.inflight.output orelse return;
|
|
||||||
var output_box: wlr.Box = undefined;
|
|
||||||
server.root.output_layout.getBox(output.wlr_output, &output_box);
|
|
||||||
|
|
||||||
const state = &self.view.inflight;
|
|
||||||
self.xwayland_surface.configure(
|
self.xwayland_surface.configure(
|
||||||
@intCast(i16, state.box.x + output_box.x),
|
@intCast(i16, inflight.box.x + output_box.x),
|
||||||
@intCast(i16, state.box.y + output_box.y),
|
@intCast(i16, inflight.box.y + output_box.y),
|
||||||
@intCast(u16, state.box.width),
|
@intCast(u16, inflight.box.width),
|
||||||
@intCast(u16, state.box.height),
|
@intCast(u16, inflight.box.height),
|
||||||
);
|
);
|
||||||
|
|
||||||
self.setActivated(state.focus != 0);
|
self.setActivated(inflight.focus != 0);
|
||||||
|
|
||||||
const fullscreen = state.output != null and state.output.?.inflight.fullscreen == self.view;
|
self.xwayland_surface.setFullscreen(output.inflight.fullscreen == self.view);
|
||||||
self.xwayland_surface.setFullscreen(fullscreen);
|
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn rootSurface(self: Self) *wlr.Surface {
|
pub fn rootSurface(self: Self) *wlr.Surface {
|
||||||
@ -237,7 +238,7 @@ fn handleRequestConfigure(
|
|||||||
self.view.pending.box.width = event.width;
|
self.view.pending.box.width = event.width;
|
||||||
self.view.pending.box.height = event.height;
|
self.view.pending.box.height = event.height;
|
||||||
}
|
}
|
||||||
self.configure();
|
server.root.applyPending();
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handleSetOverrideRedirect(
|
fn handleSetOverrideRedirect(
|
||||||
|
Loading…
x
Reference in New Issue
Block a user