river/river/Config.zig

206 lines
6.5 KiB
Zig
Raw Normal View History

2020-05-02 10:21:10 -07:00
// This file is part of river, a dynamic tiling wayland compositor.
//
// Copyright 2020 The River Developers
2020-05-02 10:21:10 -07:00
//
// 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.
2020-05-02 10:21:10 -07:00
//
// 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/>.
2020-05-02 07:27:00 -07:00
const Self = @This();
const std = @import("std");
2023-10-19 03:58:11 -07:00
const fmt = std.fmt;
const mem = std.mem;
const globber = @import("globber");
const xkb = @import("xkbcommon");
2020-05-02 07:27:00 -07:00
2023-10-19 03:58:11 -07:00
const server = &@import("main.zig").server;
const util = @import("util.zig");
const Server = @import("Server.zig");
2023-10-19 03:58:11 -07:00
const Output = @import("Output.zig");
const Mode = @import("Mode.zig");
const RuleList = @import("rule_list.zig").RuleList;
2023-10-19 03:58:11 -07:00
const View = @import("View.zig");
pub const AttachMode = enum {
top,
bottom,
};
2020-09-14 15:38:50 -07:00
pub const FocusFollowsCursorMode = enum {
disabled,
/// Only change focus on entering a surface
normal,
/// Change focus on any cursor movement
always,
2020-09-14 15:38:50 -07:00
};
pub const WarpCursorMode = enum {
disabled,
@"on-output-change",
@"on-focus-change",
};
pub const HideCursorWhenTypingMode = enum {
disabled,
enabled,
};
pub const Position = struct {
x: u31,
y: u31,
};
pub const Dimensions = struct {
width: u31,
height: u31,
};
/// Color of background in RGBA with premultiplied alpha (alpha should only affect nested sessions)
2020-08-21 10:32:21 -07:00
background_color: [4]f32 = [_]f32{ 0.0, 0.16862745, 0.21176471, 1.0 }, // Solarized base03
2020-05-02 07:27:00 -07:00
/// Width of borders in pixels
2022-11-11 11:25:21 -08:00
border_width: u31 = 2,
/// Color of border of focused window in RGBA with premultiplied alpha
2020-08-21 10:32:21 -07:00
border_color_focused: [4]f32 = [_]f32{ 0.57647059, 0.63137255, 0.63137255, 1.0 }, // Solarized base1
/// Color of border of unfocused window in RGBA with premultiplied alpha
border_color_unfocused: [4]f32 = [_]f32{ 0.34509804, 0.43137255, 0.45882353, 1.0 }, // Solarized base01
/// Color of border of urgent window in RGBA with premultiplied alpha
border_color_urgent: [4]f32 = [_]f32{ 0.86274510, 0.19607843, 0.18431373, 1.0 }, // Solarized red
2020-06-01 06:16:18 -07:00
/// Map of keymap mode name to mode id
/// Does not own the string keys. They are owned by the corresponding Mode struct.
mode_to_id: std.StringHashMap(u32),
2020-05-31 14:56:25 -07:00
2020-06-01 06:16:18 -07:00
/// All user-defined keymap modes, indexed by mode id
modes: std.ArrayListUnmanaged(Mode),
2020-05-02 07:27:00 -07:00
float_rules: RuleList(bool) = .{},
ssd_rules: RuleList(bool) = .{},
2023-06-26 17:11:13 -07:00
tag_rules: RuleList(u32) = .{},
2023-10-19 03:58:11 -07:00
output_rules: RuleList([]const u8) = .{},
position_rules: RuleList(Position) = .{},
dimensions_rules: RuleList(Dimensions) = .{},
fullscreen_rules: RuleList(bool) = .{},
2020-07-15 16:21:22 -07:00
2020-09-14 15:38:50 -07:00
/// The selected focus_follows_cursor mode
focus_follows_cursor: FocusFollowsCursorMode = .disabled,
/// If true, the cursor warps to the center of the focused output
warp_cursor: WarpCursorMode = .disabled,
/// The default layout namespace for outputs which have never had a per-output
/// value set. Call Output.handleLayoutNamespaceChange() on setting this if
/// Output.layout_namespace is null.
default_layout_namespace: []const u8 = &[0]u8{},
/// Bitmask restricting the tags of newly created views.
spawn_tagmask: u32 = std.math.maxInt(u32),
2021-06-08 00:50:42 -07:00
/// Determines where new views will be attached to the view stack.
attach_mode: AttachMode = .top,
/// Keyboard repeat rate in characters per second
repeat_rate: u31 = 25,
/// Keyboard repeat delay in milliseconds
repeat_delay: u31 = 600,
/// Cursor hide timeout in milliseconds
cursor_hide_timeout: u31 = 0,
/// Hide the cursor while typing
cursor_hide_when_typing: HideCursorWhenTypingMode = .disabled,
xkb_context: *xkb.Context,
/// The xkb keymap used for all keyboards
keymap: *xkb.Keymap,
2020-08-21 10:32:21 -07:00
pub fn init() !Self {
const xkb_context = xkb.Context.new(.no_flags) orelse return error.XkbContextFailed;
defer xkb_context.unref();
// 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(xkb_context, null, .no_flags) orelse return error.XkbKeymapFailed;
defer keymap.unref();
var self = Self{
.mode_to_id = std.StringHashMap(u32).init(util.gpa),
.modes = try std.ArrayListUnmanaged(Mode).initCapacity(util.gpa, 2),
.xkb_context = xkb_context.ref(),
.keymap = keymap.ref(),
2020-08-21 10:32:21 -07:00
};
errdefer self.deinit();
// Start with two empty modes, "normal" and "locked"
{
// Normal mode, id 0
const owned_slice = try util.gpa.dupeZ(u8, "normal");
try self.mode_to_id.putNoClobber(owned_slice, 0);
self.modes.appendAssumeCapacity(.{ .name = owned_slice });
}
{
// Locked mode, id 1
const owned_slice = try util.gpa.dupeZ(u8, "locked");
try self.mode_to_id.putNoClobber(owned_slice, 1);
self.modes.appendAssumeCapacity(.{ .name = owned_slice });
}
return self;
2020-05-02 07:27:00 -07:00
}
2020-05-16 15:03:26 -07:00
pub fn deinit(self: *Self) void {
self.mode_to_id.deinit();
for (self.modes.items) |*mode| mode.deinit();
self.modes.deinit(util.gpa);
2020-06-26 03:58:11 -07:00
self.float_rules.deinit();
self.ssd_rules.deinit();
2023-06-26 17:11:13 -07:00
self.tag_rules.deinit();
2023-10-19 03:58:11 -07:00
for (self.output_rules.rules.items) |rule| {
util.gpa.free(rule.value);
}
self.output_rules.deinit();
self.position_rules.deinit();
self.dimensions_rules.deinit();
self.fullscreen_rules.deinit();
util.gpa.free(self.default_layout_namespace);
self.keymap.unref();
self.xkb_context.unref();
}
2023-10-19 03:58:11 -07:00
pub fn outputRuleMatch(self: *Self, view: *View) !?*Output {
const output_name = self.output_rules.match(view) orelse return null;
var it = server.root.active_outputs.iterator(.forward);
while (it.next()) |output| {
const wlr_output = output.wlr_output;
if (mem.eql(u8, output_name, mem.span(wlr_output.name))) return output;
// This allows matching with "Maker Model Serial" instead of "Connector"
const maker = wlr_output.make orelse "Unknown";
const model = wlr_output.model orelse "Unknown";
const serial = wlr_output.serial orelse "Unknown";
const identifier = try fmt.allocPrint(util.gpa, "{s} {s} {s}", .{ maker, model, serial });
defer util.gpa.free(identifier);
if (mem.eql(u8, output_name, identifier)) return output;
}
return null;
}