diff options
Diffstat (limited to 'src/command/focus_view.zig')
| -rw-r--r-- | src/command/focus_view.zig | 37 |
1 files changed, 37 insertions, 0 deletions
diff --git a/src/command/focus_view.zig b/src/command/focus_view.zig new file mode 100644 index 0000000..ec51b83 --- /dev/null +++ b/src/command/focus_view.zig @@ -0,0 +1,37 @@ +const c = @import("../c.zig"); + +const Arg = @import("../command.zig").Arg; +const Seat = @import("../seat.zig").Seat; +const View = @import("../view.zig").View; +const ViewStack = @import("../view_stack.zig").ViewStack; + +/// Focus either the next or the previous visible view, depending on the enum +/// passed. Does nothing if there are 1 or 0 views in the stack. +pub fn focusView(seat: *Seat, arg: Arg) void { + const direction = arg.direction; + const output = seat.focused_output; + if (seat.focused_view) |current_focus| { + // If there is a currently focused view, focus the next visible view in the stack. + const focused_node = @fieldParentPtr(ViewStack(View).Node, "view", current_focus); + var it = switch (direction) { + .Next => ViewStack(View).iterator(focused_node, output.current_focused_tags), + .Prev => ViewStack(View).reverseIterator(focused_node, output.current_focused_tags), + }; + + // Skip past the focused node + _ = it.next(); + // Focus the next visible node if there is one + if (it.next()) |node| { + seat.focus(&node.view); + return; + } + } + + // There is either no currently focused view or the last visible view in the + // stack is focused and we need to wrap. + var it = switch (direction) { + .Next => ViewStack(View).iterator(output.views.first, output.current_focused_tags), + .Prev => ViewStack(View).reverseIterator(output.views.last, output.current_focused_tags), + }; + seat.focus(if (it.next()) |node| &node.view else null); +} |
