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 . + +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; + } +}