aboutsummaryrefslogtreecommitdiff
path: root/src/command/focus_view.zig
diff options
context:
space:
mode:
Diffstat (limited to 'src/command/focus_view.zig')
-rw-r--r--src/command/focus_view.zig37
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);
+}