2020-04-24 23:36:52 +00:00
|
|
|
{ config, pkgs, lib, ... }:
|
|
|
|
|
|
|
|
let
|
2023-04-29 16:46:19 +00:00
|
|
|
inherit (lib) mkOption mkIf types optionalString;
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
cfg = config.programs.tmux;
|
|
|
|
|
|
|
|
defaultKeyMode = "emacs";
|
|
|
|
defaultResize = 5;
|
|
|
|
defaultShortcut = "b";
|
|
|
|
defaultTerminal = "screen";
|
|
|
|
|
|
|
|
boolToStr = value: if value then "on" else "off";
|
|
|
|
|
|
|
|
tmuxConf = ''
|
|
|
|
set -g default-terminal "${cfg.terminal}"
|
|
|
|
set -g base-index ${toString cfg.baseIndex}
|
|
|
|
setw -g pane-base-index ${toString cfg.baseIndex}
|
|
|
|
|
2023-04-29 16:46:19 +00:00
|
|
|
${optionalString cfg.newSession "new-session"}
|
2020-04-24 23:36:52 +00:00
|
|
|
|
2023-04-29 16:46:19 +00:00
|
|
|
${optionalString cfg.reverseSplit ''
|
2020-04-24 23:36:52 +00:00
|
|
|
bind v split-window -h
|
|
|
|
bind s split-window -v
|
2023-04-29 16:46:19 +00:00
|
|
|
''}
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
set -g status-keys ${cfg.keyMode}
|
|
|
|
set -g mode-keys ${cfg.keyMode}
|
|
|
|
|
2023-04-29 16:46:19 +00:00
|
|
|
${optionalString (cfg.keyMode == "vi" && cfg.customPaneNavigationAndResize) ''
|
2020-04-24 23:36:52 +00:00
|
|
|
bind h select-pane -L
|
|
|
|
bind j select-pane -D
|
|
|
|
bind k select-pane -U
|
|
|
|
bind l select-pane -R
|
|
|
|
|
|
|
|
bind -r H resize-pane -L ${toString cfg.resizeAmount}
|
|
|
|
bind -r J resize-pane -D ${toString cfg.resizeAmount}
|
|
|
|
bind -r K resize-pane -U ${toString cfg.resizeAmount}
|
|
|
|
bind -r L resize-pane -R ${toString cfg.resizeAmount}
|
2023-04-29 16:46:19 +00:00
|
|
|
''}
|
2020-04-24 23:36:52 +00:00
|
|
|
|
2023-04-29 16:46:19 +00:00
|
|
|
${optionalString (cfg.shortcut != defaultShortcut) ''
|
2020-04-24 23:36:52 +00:00
|
|
|
# rebind main key: C-${cfg.shortcut}
|
|
|
|
unbind C-${defaultShortcut}
|
|
|
|
set -g prefix C-${cfg.shortcut}
|
|
|
|
bind ${cfg.shortcut} send-prefix
|
|
|
|
bind C-${cfg.shortcut} last-window
|
2023-04-29 16:46:19 +00:00
|
|
|
''}
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
setw -g aggressive-resize ${boolToStr cfg.aggressiveResize}
|
|
|
|
setw -g clock-mode-style ${if cfg.clock24 then "24" else "12"}
|
|
|
|
set -s escape-time ${toString cfg.escapeTime}
|
|
|
|
set -g history-limit ${toString cfg.historyLimit}
|
|
|
|
|
2023-10-09 19:29:22 +00:00
|
|
|
${cfg.extraConfigBeforePlugins}
|
|
|
|
|
2022-01-27 00:19:43 +00:00
|
|
|
${lib.optionalString (cfg.plugins != []) ''
|
|
|
|
# Run plugins
|
|
|
|
${lib.concatMapStringsSep "\n" (x: "run-shell ${x.rtp}") cfg.plugins}
|
|
|
|
|
|
|
|
''}
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
${cfg.extraConfig}
|
|
|
|
'';
|
|
|
|
|
|
|
|
in {
|
|
|
|
###### interface
|
|
|
|
|
|
|
|
options = {
|
|
|
|
programs.tmux = {
|
|
|
|
|
|
|
|
enable = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Whenever to configure {command}`tmux` system-wide.";
|
2020-04-24 23:36:52 +00:00
|
|
|
relatedPackages = [ "tmux" ];
|
|
|
|
};
|
|
|
|
|
|
|
|
aggressiveResize = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Resize the window to the size of the smallest session for which it is the current window.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
baseIndex = mkOption {
|
|
|
|
default = 0;
|
|
|
|
example = 1;
|
|
|
|
type = types.int;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Base index for windows and panes.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
clock24 = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Use 24 hour clock.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
customPaneNavigationAndResize = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Override the hjkl and HJKL bindings for pane navigation and resizing in VI mode.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
escapeTime = mkOption {
|
|
|
|
default = 500;
|
|
|
|
example = 0;
|
|
|
|
type = types.int;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Time in milliseconds for which tmux waits after an escape is input.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
2023-10-09 19:29:22 +00:00
|
|
|
extraConfigBeforePlugins = mkOption {
|
|
|
|
default = "";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2023-10-09 19:29:22 +00:00
|
|
|
Additional contents of /etc/tmux.conf, to be run before sourcing plugins.
|
|
|
|
'';
|
|
|
|
type = types.lines;
|
|
|
|
};
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
extraConfig = mkOption {
|
|
|
|
default = "";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2023-10-09 19:29:22 +00:00
|
|
|
Additional contents of /etc/tmux.conf, to be run after sourcing plugins.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
type = types.lines;
|
|
|
|
};
|
|
|
|
|
|
|
|
historyLimit = mkOption {
|
|
|
|
default = 2000;
|
|
|
|
example = 5000;
|
|
|
|
type = types.int;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Maximum number of lines held in window history.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
keyMode = mkOption {
|
|
|
|
default = defaultKeyMode;
|
|
|
|
example = "vi";
|
|
|
|
type = types.enum [ "emacs" "vi" ];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "VI or Emacs style shortcuts.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
newSession = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Automatically spawn a session if trying to attach and none are running.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
reverseSplit = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Reverse the window split shortcuts.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
resizeAmount = mkOption {
|
|
|
|
default = defaultResize;
|
|
|
|
example = 10;
|
|
|
|
type = types.int;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Number of lines/columns when resizing.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
shortcut = mkOption {
|
|
|
|
default = defaultShortcut;
|
|
|
|
example = "a";
|
|
|
|
type = types.str;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Ctrl following by this key is used as the main shortcut.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
terminal = mkOption {
|
|
|
|
default = defaultTerminal;
|
|
|
|
example = "screen-256color";
|
|
|
|
type = types.str;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2023-04-29 16:46:19 +00:00
|
|
|
Set the $TERM variable. Use tmux-direct if italics or 24bit true color
|
|
|
|
support is needed.
|
|
|
|
'';
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
secureSocket = mkOption {
|
|
|
|
default = true;
|
|
|
|
type = types.bool;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Store tmux socket under /run, which is more secure than /tmp, but as a
|
|
|
|
downside it doesn't survive user logout.
|
|
|
|
'';
|
|
|
|
};
|
2022-01-27 00:19:43 +00:00
|
|
|
|
|
|
|
plugins = mkOption {
|
|
|
|
default = [];
|
|
|
|
type = types.listOf types.package;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "List of plugins to install.";
|
2022-01-27 00:19:43 +00:00
|
|
|
example = lib.literalExpression "[ pkgs.tmuxPlugins.nord ]";
|
|
|
|
};
|
2022-11-21 17:40:18 +00:00
|
|
|
|
|
|
|
withUtempter = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-11-21 17:40:18 +00:00
|
|
|
Whether to enable libutempter for tmux.
|
|
|
|
This is required so that tmux can write to /var/run/utmp (which can be queried with `who` to display currently connected user sessions).
|
|
|
|
Note, this will add a guid wrapper for the group utmp!
|
|
|
|
'';
|
|
|
|
default = true;
|
|
|
|
type = types.bool;
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
###### implementation
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
environment = {
|
|
|
|
etc."tmux.conf".text = tmuxConf;
|
|
|
|
|
2022-01-27 00:19:43 +00:00
|
|
|
systemPackages = [ pkgs.tmux ] ++ cfg.plugins;
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
variables = {
|
|
|
|
TMUX_TMPDIR = lib.optional cfg.secureSocket ''''${XDG_RUNTIME_DIR:-"/run/user/$(id -u)"}'';
|
|
|
|
};
|
|
|
|
};
|
2022-11-21 17:40:18 +00:00
|
|
|
security.wrappers = mkIf cfg.withUtempter {
|
|
|
|
utempter = {
|
|
|
|
source = "${pkgs.libutempter}/lib/utempter/utempter";
|
|
|
|
owner = "root";
|
|
|
|
group = "utmp";
|
|
|
|
setuid = false;
|
|
|
|
setgid = true;
|
|
|
|
};
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
imports = [
|
|
|
|
(lib.mkRenamedOptionModule [ "programs" "tmux" "extraTmuxConf" ] [ "programs" "tmux" "extraConfig" ])
|
|
|
|
];
|
|
|
|
}
|