2021-05-23 08:10:26 -07:00
|
|
|
// This file is part of river, a dynamic tiling wayland compositor.
|
|
|
|
//
|
|
|
|
// Copyright 2021 The River Developers
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
const Subsurface = @This();
|
|
|
|
|
|
|
|
const std = @import("std");
|
|
|
|
const wlr = @import("wlroots");
|
|
|
|
const wl = @import("wayland").server.wl;
|
|
|
|
|
2021-06-07 11:34:54 -07:00
|
|
|
const server = &@import("main.zig").server;
|
2021-05-23 08:10:26 -07:00
|
|
|
const util = @import("util.zig");
|
|
|
|
|
2021-06-07 11:34:54 -07:00
|
|
|
const DragIcon = @import("DragIcon.zig");
|
2021-05-23 08:10:26 -07:00
|
|
|
const LayerSurface = @import("LayerSurface.zig");
|
|
|
|
const View = @import("View.zig");
|
|
|
|
|
|
|
|
pub const Parent = union(enum) {
|
|
|
|
view: *View,
|
|
|
|
layer_surface: *LayerSurface,
|
2021-06-07 11:34:54 -07:00
|
|
|
drag_icon: *DragIcon,
|
2021-05-23 08:10:26 -07:00
|
|
|
|
|
|
|
pub fn damageWholeOutput(parent: Parent) void {
|
|
|
|
switch (parent) {
|
|
|
|
.view => |view| view.output.damage.addWhole(),
|
|
|
|
.layer_surface => |layer_surface| layer_surface.output.damage.addWhole(),
|
2021-06-07 11:34:54 -07:00
|
|
|
.drag_icon => |drag_icon| {
|
|
|
|
var it = server.root.outputs.first;
|
|
|
|
while (it) |node| : (it = node.next) node.data.damage.addWhole();
|
|
|
|
},
|
2021-05-23 08:10:26 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/// The parent at the root of this surface tree
|
|
|
|
parent: Parent,
|
|
|
|
wlr_subsurface: *wlr.Subsurface,
|
|
|
|
|
|
|
|
// Always active
|
|
|
|
destroy: wl.Listener(*wlr.Subsurface) = wl.Listener(*wlr.Subsurface).init(handleDestroy),
|
|
|
|
map: wl.Listener(*wlr.Subsurface) = wl.Listener(*wlr.Subsurface).init(handleMap),
|
|
|
|
unmap: wl.Listener(*wlr.Subsurface) = wl.Listener(*wlr.Subsurface).init(handleUnmap),
|
|
|
|
new_subsurface: wl.Listener(*wlr.Subsurface) = wl.Listener(*wlr.Subsurface).init(handleNewSubsurface),
|
|
|
|
|
|
|
|
// Only active while mapped
|
|
|
|
commit: wl.Listener(*wlr.Surface) = wl.Listener(*wlr.Surface).init(handleCommit),
|
|
|
|
|
|
|
|
pub fn create(wlr_subsurface: *wlr.Subsurface, parent: Parent) void {
|
|
|
|
const subsurface = util.gpa.create(Subsurface) catch {
|
|
|
|
std.log.crit("out of memory", .{});
|
|
|
|
wlr_subsurface.resource.getClient().postNoMemory();
|
|
|
|
return;
|
|
|
|
};
|
|
|
|
subsurface.* = .{ .wlr_subsurface = wlr_subsurface, .parent = parent };
|
|
|
|
|
|
|
|
wlr_subsurface.events.destroy.add(&subsurface.destroy);
|
|
|
|
wlr_subsurface.events.map.add(&subsurface.map);
|
|
|
|
wlr_subsurface.events.unmap.add(&subsurface.unmap);
|
|
|
|
wlr_subsurface.surface.events.new_subsurface.add(&subsurface.new_subsurface);
|
2021-06-05 12:30:43 -07:00
|
|
|
|
2021-06-14 13:27:08 -07:00
|
|
|
Subsurface.handleExisting(wlr_subsurface.surface, parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create Subsurface structs to track subsurfaces already present on the
|
|
|
|
/// given surface when river becomes aware of the surface as we won't
|
|
|
|
/// recieve a new_subsurface event for them.
|
|
|
|
pub fn handleExisting(surface: *wlr.Surface, parent: Parent) void {
|
|
|
|
var below_it = surface.subsurfaces_below.iterator(.forward);
|
|
|
|
while (below_it.next()) |s| Subsurface.create(s, parent);
|
|
|
|
|
|
|
|
var above_it = surface.subsurfaces_above.iterator(.forward);
|
|
|
|
while (above_it.next()) |s| Subsurface.create(s, parent);
|
2021-05-23 08:10:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn handleDestroy(listener: *wl.Listener(*wlr.Subsurface), wlr_subsurface: *wlr.Subsurface) void {
|
|
|
|
const subsurface = @fieldParentPtr(Subsurface, "destroy", listener);
|
|
|
|
|
|
|
|
subsurface.destroy.link.remove();
|
|
|
|
subsurface.map.link.remove();
|
|
|
|
subsurface.unmap.link.remove();
|
|
|
|
subsurface.new_subsurface.link.remove();
|
|
|
|
|
|
|
|
util.gpa.destroy(subsurface);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn handleMap(listener: *wl.Listener(*wlr.Subsurface), wlr_subsurface: *wlr.Subsurface) void {
|
|
|
|
const subsurface = @fieldParentPtr(Subsurface, "map", listener);
|
|
|
|
|
|
|
|
wlr_subsurface.surface.events.commit.add(&subsurface.commit);
|
|
|
|
subsurface.parent.damageWholeOutput();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn handleUnmap(listener: *wl.Listener(*wlr.Subsurface), wlr_subsurface: *wlr.Subsurface) void {
|
|
|
|
const subsurface = @fieldParentPtr(Subsurface, "unmap", listener);
|
|
|
|
|
|
|
|
subsurface.commit.link.remove();
|
|
|
|
subsurface.parent.damageWholeOutput();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn handleCommit(listener: *wl.Listener(*wlr.Surface), surface: *wlr.Surface) void {
|
|
|
|
const subsurface = @fieldParentPtr(Subsurface, "commit", listener);
|
|
|
|
|
|
|
|
subsurface.parent.damageWholeOutput();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn handleNewSubsurface(listener: *wl.Listener(*wlr.Subsurface), new_wlr_subsurface: *wlr.Subsurface) void {
|
|
|
|
const subsurface = @fieldParentPtr(Subsurface, "new_subsurface", listener);
|
|
|
|
|
|
|
|
Subsurface.create(new_wlr_subsurface, subsurface.parent);
|
|
|
|
}
|