aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--doc/riverctl.1.scd5
-rw-r--r--river/Config.zig13
-rw-r--r--river/Keyboard.zig3
-rw-r--r--river/command.zig1
-rw-r--r--river/command/keyboard.zig88
5 files changed, 109 insertions, 1 deletions
diff --git a/doc/riverctl.1.scd b/doc/riverctl.1.scd
index aaddd9b..177cb32 100644
--- a/doc/riverctl.1.scd
+++ b/doc/riverctl.1.scd
@@ -334,6 +334,11 @@ A complete list may be found in _/usr/include/linux/input-event-codes.h_
*list-input-configs*
List all input configurations.
+*keyboard-layout* _layout_ [-variant _variant_] [-model _model_] [-options _options_] [-rules _rules_]
+ Set the XKB layout for all keyboards. All values other than the layout
+ name are optional. XKB will fill in unspecified values based on
+ heuristics and the environment. Duplicate flags are not allowed.
+
*keyboard-group-create* _group_name_
Create a keyboard group. A keyboard group collects multiple keyboards in
a single logical keyboard. This means that all state, like the active
diff --git a/river/Config.zig b/river/Config.zig
index b105a4e..74b8553 100644
--- a/river/Config.zig
+++ b/river/Config.zig
@@ -17,6 +17,8 @@
const Self = @This();
const std = @import("std");
+const mem = std.mem;
+const xkb = @import("xkbcommon");
const util = @import("util.zig");
@@ -100,6 +102,9 @@ cursor_hide_timeout: u31 = 0,
/// Hide the cursor while typing
cursor_hide_when_typing: HideCursorWhenTypingMode = .disabled,
+/// Keyboard layout configuration
+keyboard_layout: ?xkb.RuleNames = null,
+
pub fn init() !Self {
var self = Self{
.mode_to_id = std.StringHashMap(u32).init(util.gpa),
@@ -129,6 +134,14 @@ pub fn deinit(self: *Self) void {
for (self.modes.items) |*mode| mode.deinit();
self.modes.deinit(util.gpa);
+ if (self.keyboard_layout) |kl| {
+ if (kl.rules) |s| util.gpa.free(mem.span(s));
+ if (kl.model) |s| util.gpa.free(mem.span(s));
+ if (kl.layout) |s| util.gpa.free(mem.span(s));
+ if (kl.variant) |s| util.gpa.free(mem.span(s));
+ if (kl.options) |s| util.gpa.free(mem.span(s));
+ }
+
{
var it = self.float_filter_app_ids.keyIterator();
while (it.next()) |key| util.gpa.free(key.*);
diff --git a/river/Keyboard.zig b/river/Keyboard.zig
index 22f3d0b..4827ffa 100644
--- a/river/Keyboard.zig
+++ b/river/Keyboard.zig
@@ -51,7 +51,8 @@ pub fn init(self: *Self, seat: *Seat, wlr_device: *wlr.InputDevice) !void {
// Passing null here indicates that defaults from libxkbcommon and
// its XKB_DEFAULT_LAYOUT, XKB_DEFAULT_OPTIONS, etc. should be used.
- const keymap = xkb.Keymap.newFromNames(context, null, .no_flags) orelse return error.XkbKeymapFailed;
+ const layout_config = if (server.config.keyboard_layout) |kl| &kl else null;
+ const keymap = xkb.Keymap.newFromNames(context, layout_config, .no_flags) orelse return error.XkbKeymapFailed;
defer keymap.unref();
const wlr_keyboard = self.device.wlr_device.toKeyboard();
diff --git a/river/command.zig b/river/command.zig
index 55dd7ca..0515963 100644
--- a/river/command.zig
+++ b/river/command.zig
@@ -89,6 +89,7 @@ const command_impls = std.ComptimeStringMap(
.{ "unmap-switch", @import("command/map.zig").unmapSwitch },
.{ "xcursor-theme", @import("command/xcursor_theme.zig").xcursorTheme },
.{ "zoom", @import("command/zoom.zig").zoom },
+ .{ "keyboard-layout", @import("command/keyboard.zig").keyboardLayout },
.{ "keyboard-group-create", @import("command/keyboard_group.zig").keyboardGroupCreate },
.{ "keyboard-group-destroy", @import("command/keyboard_group.zig").keyboardGroupDestroy },
.{ "keyboard-group-add", @import("command/keyboard_group.zig").keyboardGroupAdd },
diff --git a/river/command/keyboard.zig b/river/command/keyboard.zig
new file mode 100644
index 0000000..840e171
--- /dev/null
+++ b/river/command/keyboard.zig
@@ -0,0 +1,88 @@
+// This file is part of river, a dynamic tiling wayland compositor.
+//
+// Copyright 2022 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, version 3.
+//
+// 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 std = @import("std");
+const mem = std.mem;
+
+const xkb = @import("xkbcommon");
+
+const server = &@import("../main.zig").server;
+const util = @import("../util.zig");
+
+const Error = @import("../command.zig").Error;
+const Seat = @import("../Seat.zig");
+
+pub fn keyboardLayout(
+ _: *Seat,
+ args: []const [:0]const u8,
+ _: *?[]const u8,
+) Error!void {
+ if (args.len < 2) return Error.NotEnoughArguments;
+ if (args.len % 2 != 0) return Error.InvalidValue;
+
+ // Do not carry over any previous keyboard layout configuration, always
+ // start fresh.
+ if (server.config.keyboard_layout) |kl| {
+ if (kl.rules) |s| util.gpa.free(mem.span(s));
+ if (kl.model) |s| util.gpa.free(mem.span(s));
+ if (kl.layout) |s| util.gpa.free(mem.span(s));
+ if (kl.variant) |s| util.gpa.free(mem.span(s));
+ if (kl.options) |s| util.gpa.free(mem.span(s));
+ }
+
+ server.config.keyboard_layout = xkb.RuleNames{
+ .layout = try util.gpa.dupeZ(u8, args[1]),
+ .rules = null,
+ .model = null,
+ .variant = null,
+ .options = null,
+ };
+
+ // TODO[zig]: this can be solved more elegantly with an inline for loop, but
+ // on version 0.9.1 that crashes the compiler.
+ var i: usize = 2;
+ while (i < args.len - 1) : (i += 2) {
+ if (mem.eql(u8, args[i], "-variant")) {
+ // Do not allow duplicate flags.
+ if (server.config.keyboard_layout.?.variant != null) return error.InvalidValue;
+ server.config.keyboard_layout.?.variant = try util.gpa.dupeZ(u8, args[i + 1]);
+ } else if (mem.eql(u8, args[i], "-model")) {
+ if (server.config.keyboard_layout.?.model != null) return error.InvalidValue;
+ server.config.keyboard_layout.?.model = try util.gpa.dupeZ(u8, args[i + 1]);
+ } else if (mem.eql(u8, args[i], "-options")) {
+ if (server.config.keyboard_layout.?.options != null) return error.InvalidValue;
+ server.config.keyboard_layout.?.options = try util.gpa.dupeZ(u8, args[i + 1]);
+ } else if (mem.eql(u8, args[i], "-rules")) {
+ if (server.config.keyboard_layout.?.rules != null) return error.InvalidValue;
+ server.config.keyboard_layout.?.rules = try util.gpa.dupeZ(u8, args[i + 1]);
+ } else {
+ return error.InvalidValue;
+ }
+ }
+
+ const context = xkb.Context.new(.no_flags) orelse return error.OutOfMemory;
+ defer context.unref();
+
+ const keymap = xkb.Keymap.newFromNames(context, &server.config.keyboard_layout.?, .no_flags) orelse return error.InvalidValue;
+ defer keymap.unref();
+
+ var it = server.input_manager.devices.iterator(.forward);
+ while (it.next()) |device| {
+ if (device.wlr_device.type != .keyboard) continue;
+ const wlr_keyboard = device.wlr_device.toKeyboard();
+ if (!wlr_keyboard.setKeymap(keymap)) return error.OutOfMemory;
+ }
+}