2018-11-19 17:50:35 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
# TODO: Re-write tests to support profiles.
|
|
|
|
|
2018-11-19 17:50:35 +01:00
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
|
|
|
|
cfg = config.programs.vscode;
|
|
|
|
|
2019-07-29 22:39:25 +02:00
|
|
|
vscodePname = cfg.package.pname;
|
2023-02-08 11:00:27 +01:00
|
|
|
vscodeVersion = cfg.package.version;
|
2019-07-29 22:39:25 +02:00
|
|
|
|
2020-11-30 03:54:55 +01:00
|
|
|
jsonFormat = pkgs.formats.json { };
|
|
|
|
|
2019-07-29 22:39:25 +02:00
|
|
|
configDir = {
|
|
|
|
"vscode" = "Code";
|
|
|
|
"vscode-insiders" = "Code - Insiders";
|
2019-11-11 17:50:34 +01:00
|
|
|
"vscodium" = "VSCodium";
|
2024-02-05 07:34:43 +01:00
|
|
|
"openvscode-server" = "OpenVSCode Server";
|
2019-07-29 22:39:25 +02:00
|
|
|
}.${vscodePname};
|
|
|
|
|
2019-12-06 14:00:28 +01:00
|
|
|
extensionDir = {
|
|
|
|
"vscode" = "vscode";
|
|
|
|
"vscode-insiders" = "vscode-insiders";
|
|
|
|
"vscodium" = "vscode-oss";
|
2024-02-05 07:34:43 +01:00
|
|
|
"openvscode-server" = "openvscode-server";
|
2019-12-06 14:00:28 +01:00
|
|
|
}.${vscodePname};
|
|
|
|
|
2020-10-12 22:51:12 +02:00
|
|
|
userDir = if pkgs.stdenv.hostPlatform.isDarwin then
|
|
|
|
"Library/Application Support/${configDir}/User"
|
|
|
|
else
|
|
|
|
"${config.xdg.configHome}/${configDir}/User";
|
2020-06-22 20:48:22 +02:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
configFilePath = name:
|
|
|
|
"${userDir}/${
|
|
|
|
optionalString (name != "default") "profiles/${name}/"
|
|
|
|
}settings.json";
|
|
|
|
tasksFilePath = name:
|
|
|
|
"${userDir}/${
|
|
|
|
optionalString (name != "default") "profiles/${name}/"
|
|
|
|
}tasks.json";
|
|
|
|
keybindingsFilePath = name:
|
|
|
|
"${userDir}/${
|
|
|
|
optionalString (name != "default") "profiles/${name}/"
|
|
|
|
}keybindings.json";
|
|
|
|
|
|
|
|
snippetDir = name:
|
|
|
|
"${userDir}/${
|
|
|
|
optionalString (name != "default") "profiles/${name}/"
|
|
|
|
}snippets";
|
2023-03-13 18:45:03 +01:00
|
|
|
|
2019-07-29 22:39:25 +02:00
|
|
|
# TODO: On Darwin where are the extensions?
|
2019-12-06 14:00:28 +01:00
|
|
|
extensionPath = ".${extensionDir}/extensions";
|
2018-11-19 17:50:35 +01:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
extensionJson = ext: pkgs.vscode-utils.toExtensionJson ext;
|
|
|
|
extensionJsonFile = name: text:
|
|
|
|
pkgs.writeTextFile {
|
|
|
|
inherit text;
|
|
|
|
name = "extensions-json-${name}";
|
|
|
|
destination = "/share/vscode/extensions/extensions.json";
|
|
|
|
};
|
2023-02-08 11:00:27 +01:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
mergedUserSettings = userSettings:
|
|
|
|
userSettings
|
2022-05-17 22:39:11 +02:00
|
|
|
// optionalAttrs (!cfg.enableUpdateCheck) { "update.mode" = "none"; }
|
|
|
|
// optionalAttrs (!cfg.enableExtensionUpdateCheck) {
|
|
|
|
"extensions.autoCheckUpdates" = false;
|
|
|
|
};
|
2022-02-07 18:40:59 +01:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
profileType = default:
|
|
|
|
types.submodule {
|
|
|
|
options = {
|
|
|
|
userSettings = mkOption {
|
|
|
|
type = jsonFormat.type;
|
|
|
|
default = { };
|
|
|
|
example = literalExpression ''
|
|
|
|
{
|
|
|
|
"files.autoSave" = "off";
|
|
|
|
"[nix]"."editor.tabSize" = 2;
|
|
|
|
}
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
Configuration written to Visual Studio Code's
|
|
|
|
{file}`settings.json`.
|
|
|
|
'';
|
|
|
|
};
|
2022-05-17 22:39:11 +02:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
userTasks = mkOption {
|
|
|
|
type = jsonFormat.type;
|
|
|
|
default = { };
|
|
|
|
example = literalExpression ''
|
|
|
|
{
|
|
|
|
version = "2.0.0";
|
|
|
|
tasks = [
|
|
|
|
{
|
|
|
|
type = "shell";
|
|
|
|
label = "Hello task";
|
|
|
|
command = "hello";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
}
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
Configuration written to Visual Studio Code's
|
|
|
|
{file}`tasks.json`.
|
|
|
|
'';
|
|
|
|
};
|
2018-11-19 17:50:35 +01:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
keybindings = mkOption {
|
|
|
|
type = types.listOf (types.submodule {
|
|
|
|
options = {
|
|
|
|
key = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
example = "ctrl+c";
|
|
|
|
description = "The key or key-combination to bind.";
|
|
|
|
};
|
|
|
|
|
|
|
|
command = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
example = "editor.action.clipboardCopyAction";
|
|
|
|
description = "The VS Code command to execute.";
|
|
|
|
};
|
|
|
|
|
|
|
|
when = mkOption {
|
|
|
|
type = types.nullOr (types.str);
|
|
|
|
default = null;
|
|
|
|
example = "textInputFocus";
|
|
|
|
description = "Optional context filter.";
|
|
|
|
};
|
|
|
|
|
|
|
|
# https://code.visualstudio.com/docs/getstarted/keybindings#_command-arguments
|
|
|
|
args = mkOption {
|
|
|
|
type = types.nullOr (jsonFormat.type);
|
|
|
|
default = null;
|
|
|
|
example = { direction = "up"; };
|
|
|
|
description = "Optional arguments for a command.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
});
|
|
|
|
default = [ ];
|
|
|
|
example = literalExpression ''
|
|
|
|
[
|
2022-03-18 23:54:24 +01:00
|
|
|
{
|
2024-07-16 14:09:45 +02:00
|
|
|
key = "ctrl+c";
|
|
|
|
command = "editor.action.clipboardCopyAction";
|
|
|
|
when = "textInputFocus";
|
2022-03-18 23:54:24 +01:00
|
|
|
}
|
2024-07-16 14:09:45 +02:00
|
|
|
]
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
Keybindings written to Visual Studio Code's
|
|
|
|
{file}`keybindings.json`.
|
|
|
|
'';
|
|
|
|
};
|
2020-06-22 20:48:22 +02:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
extensions = mkOption {
|
|
|
|
type = types.listOf types.package;
|
|
|
|
default = [ ];
|
|
|
|
example = literalExpression "[ pkgs.vscode-extensions.bbenoist.nix ]";
|
|
|
|
description = ''
|
|
|
|
The extensions Visual Studio Code should be started with.
|
|
|
|
'';
|
|
|
|
};
|
2021-01-30 01:00:58 +01:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
languageSnippets = mkOption {
|
|
|
|
type = jsonFormat.type;
|
|
|
|
default = { };
|
|
|
|
example = {
|
|
|
|
haskell = {
|
|
|
|
fixme = {
|
|
|
|
prefix = [ "fixme" ];
|
|
|
|
body = [ "$LINE_COMMENT FIXME: $0" ];
|
|
|
|
description = "Insert a FIXME remark";
|
|
|
|
};
|
2021-01-30 01:00:58 +01:00
|
|
|
};
|
2020-06-22 20:48:22 +02:00
|
|
|
};
|
2024-07-16 14:09:45 +02:00
|
|
|
description = "Defines user snippets for different languages.";
|
|
|
|
};
|
2023-03-13 18:45:03 +01:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
globalSnippets = mkOption {
|
|
|
|
type = jsonFormat.type;
|
|
|
|
default = { };
|
|
|
|
example = {
|
2023-03-13 18:45:03 +01:00
|
|
|
fixme = {
|
|
|
|
prefix = [ "fixme" ];
|
|
|
|
body = [ "$LINE_COMMENT FIXME: $0" ];
|
|
|
|
description = "Insert a FIXME remark";
|
|
|
|
};
|
|
|
|
};
|
2024-07-16 14:09:45 +02:00
|
|
|
description = "Defines global user snippets.";
|
2023-03-13 18:45:03 +01:00
|
|
|
};
|
2024-07-16 14:09:45 +02:00
|
|
|
} // optionalAttrs default {
|
|
|
|
name = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
description = "Visual Studio Code's Profile name.";
|
2023-03-13 18:45:03 +01:00
|
|
|
};
|
|
|
|
};
|
2018-11-19 17:50:35 +01:00
|
|
|
};
|
2024-07-16 14:09:45 +02:00
|
|
|
allProfiles = cfg.profiles ++ [ cfg.defaultProfile ];
|
|
|
|
in {
|
|
|
|
imports = [
|
|
|
|
(mkChangedOptionModule [ "programs" "vscode" "immutableExtensionsDir" ] [
|
|
|
|
"programs"
|
|
|
|
"vscode"
|
|
|
|
"mutableExtensionsDir"
|
|
|
|
] (config: !config.programs.vscode.immutableExtensionsDir))
|
2024-07-16 14:10:39 +02:00
|
|
|
] ++ map (v:
|
|
|
|
mkRenamedOptionModule [ "programs" "vscode" v ] [
|
|
|
|
"programs"
|
|
|
|
"vscode"
|
|
|
|
"defaultProfile"
|
|
|
|
v
|
|
|
|
]) [
|
|
|
|
"userSettings"
|
|
|
|
"userTasks"
|
|
|
|
"keybindings"
|
|
|
|
"extensions"
|
|
|
|
"languageSnippets"
|
|
|
|
"globalSnippets"
|
|
|
|
];
|
2024-07-16 14:09:45 +02:00
|
|
|
|
|
|
|
options.programs.vscode = {
|
|
|
|
enable = mkEnableOption "Visual Studio Code";
|
|
|
|
|
|
|
|
package = mkOption {
|
|
|
|
type = types.package;
|
|
|
|
default = pkgs.vscode;
|
|
|
|
defaultText = literalExpression "pkgs.vscode";
|
|
|
|
example = literalExpression "pkgs.vscodium";
|
|
|
|
description = ''
|
|
|
|
Version of Visual Studio Code to install.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
enableUpdateCheck = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
description = ''
|
|
|
|
Whether to enable update checks/notifications.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
enableExtensionUpdateCheck = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
description = ''
|
|
|
|
Whether to enable update notifications for extensions.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
mutableExtensionsDir = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = cfg.profiles == [ ];
|
|
|
|
example = false;
|
|
|
|
description = ''
|
|
|
|
Whether extensions can be installed or updated manually
|
|
|
|
or by Visual Studio Code. This option is effective only
|
|
|
|
when there is a single profile (i.e. default).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
profiles = mkOption {
|
|
|
|
type = types.listOf (profileType true);
|
|
|
|
default = [ ];
|
|
|
|
};
|
|
|
|
defaultProfile = mkOption {
|
|
|
|
type = profileType false;
|
|
|
|
default = { };
|
|
|
|
};
|
2018-11-19 17:50:35 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
2024-07-16 14:09:45 +02:00
|
|
|
warnings = [
|
|
|
|
(mkIf (cfg.profiles != [ ] && cfg.mutableExtensionsDir)
|
|
|
|
"programs.vscode.mutableExtensionsDir can be used only if profiles is an empty list.")
|
|
|
|
];
|
|
|
|
|
2019-07-29 22:39:25 +02:00
|
|
|
home.packages = [ cfg.package ];
|
2018-11-19 17:50:35 +01:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
/* *
|
|
|
|
TODO: Write a home.activation script for ${userDir}/globalStorage/storage.json, appending
|
|
|
|
every profile in the format `{ "name": <profile_name>, "location": <profile_name> }` to the
|
|
|
|
userDataProfiles array.
|
|
|
|
|
|
|
|
This file needs to mutable, and cannot be symlinked. This is because the file stores other data,
|
|
|
|
such as background themes, keybindingReferences, etc.
|
|
|
|
*/
|
|
|
|
|
|
|
|
home.file = mkMerge (flatten [
|
|
|
|
(map (v:
|
|
|
|
let
|
|
|
|
# The default profile does not have the `name` key
|
|
|
|
name = if v ? name then v.name else "default";
|
|
|
|
in [
|
|
|
|
(mkIf ((mergedUserSettings v.userSettings) != { }) {
|
|
|
|
"${configFilePath name}".source =
|
|
|
|
jsonFormat.generate "vscode-user-settings"
|
|
|
|
(mergedUserSettings v.userSettings);
|
|
|
|
})
|
2022-02-02 21:57:11 +01:00
|
|
|
|
2024-07-16 14:09:45 +02:00
|
|
|
(mkIf (v.userTasks != { }) {
|
|
|
|
"${tasksFilePath name}".source =
|
|
|
|
jsonFormat.generate "vscode-user-tasks" v.userTasks;
|
|
|
|
})
|
|
|
|
|
|
|
|
(mkIf (v.keybindings != [ ]) {
|
|
|
|
"${keybindingsFilePath name}".source =
|
|
|
|
jsonFormat.generate "vscode-keybindings"
|
|
|
|
(map (filterAttrs (_: v: v != null)) v.keybindings);
|
|
|
|
})
|
|
|
|
|
|
|
|
(mkIf (v.languageSnippets != { }) (lib.mapAttrs' (language: snippet:
|
|
|
|
lib.nameValuePair "${snippetDir name}/${language}.json" {
|
|
|
|
source =
|
|
|
|
jsonFormat.generate "user-snippet-${language}.json" snippet;
|
|
|
|
}) v.languageSnippets))
|
|
|
|
|
|
|
|
(mkIf (v.globalSnippets != { }) {
|
|
|
|
"${snippetDir name}/global.code-snippets".source =
|
|
|
|
jsonFormat.generate "user-snippet-global.code-snippets"
|
|
|
|
v.globalSnippets;
|
|
|
|
})
|
|
|
|
]) allProfiles)
|
|
|
|
|
|
|
|
# We write extensions.json for all profiles, except the default profile,
|
|
|
|
# since that is handled by code below.
|
|
|
|
(mkIf (cfg.profiles != [ ]) (listToAttrs (map (v:
|
|
|
|
nameValuePair "${userDir}/profiles/${v.name}/extensions.json" {
|
|
|
|
source = "${
|
|
|
|
extensionJsonFile v.name (extensionJson v.extensions)
|
|
|
|
}/share/vscode/extensions/extensions.json";
|
|
|
|
}) cfg.profiles)))
|
|
|
|
|
|
|
|
(mkIf ((filter (v: v.extensions != [ ]) allProfiles) != [ ]) (let
|
2022-02-02 21:57:11 +01:00
|
|
|
# Adapted from https://discourse.nixos.org/t/vscode-extensions-setup/1801/2
|
2024-07-16 14:09:45 +02:00
|
|
|
subDir = "share/vscode/extensions";
|
2022-07-15 16:29:10 +02:00
|
|
|
toPaths = ext:
|
|
|
|
map (k: { "${extensionPath}/${k}".source = "${ext}/${subDir}/${k}"; })
|
|
|
|
(if ext ? vscodeExtUniqueId then
|
|
|
|
[ ext.vscodeExtUniqueId ]
|
|
|
|
else
|
|
|
|
builtins.attrNames (builtins.readDir (ext + "/${subDir}")));
|
2024-07-16 14:09:45 +02:00
|
|
|
in if (cfg.mutableExtensionsDir && cfg.profiles == [ ]) then
|
|
|
|
mkMerge (concatMap toPaths (flatten (map (v: v.extensions) allProfiles))
|
2023-02-08 12:18:28 +01:00
|
|
|
++ lib.optional (lib.versionAtLeast vscodeVersion "1.74.0") {
|
2023-02-08 11:00:27 +01:00
|
|
|
# Whenever our immutable extensions.json changes, force VSCode to regenerate
|
|
|
|
# extensions.json with both mutable and immutable extensions.
|
|
|
|
"${extensionPath}/.extensions-immutable.json" = {
|
2024-07-16 14:09:45 +02:00
|
|
|
text = extensionJson cfg.defaultProfile.extensions;
|
2023-02-08 11:00:27 +01:00
|
|
|
onChange = ''
|
2024-01-13 23:15:00 +01:00
|
|
|
run rm $VERBOSE_ARG -f ${extensionPath}/{extensions.json,.init-default-profile-extensions}
|
2024-01-23 22:59:26 +01:00
|
|
|
verboseEcho "Regenerating VSCode extensions.json"
|
2024-01-13 23:15:00 +01:00
|
|
|
run ${getExe cfg.package} --list-extensions > /dev/null
|
2023-02-08 11:00:27 +01:00
|
|
|
'';
|
|
|
|
};
|
2023-02-08 12:18:28 +01:00
|
|
|
})
|
2022-02-07 18:40:59 +01:00
|
|
|
else {
|
2022-07-15 16:29:10 +02:00
|
|
|
"${extensionPath}".source = let
|
|
|
|
combinedExtensionsDrv = pkgs.buildEnv {
|
|
|
|
name = "vscode-extensions";
|
2024-07-16 14:09:45 +02:00
|
|
|
paths = flatten (map (v: v.extensions) allProfiles) ++ lib.optional
|
|
|
|
(lib.versionAtLeast vscodeVersion "1.74.0" && cfg.defaultProfile
|
|
|
|
!= { }) (extensionJsonFile "default"
|
|
|
|
(extensionJson cfg.defaultProfile.extensions));
|
2022-07-15 16:29:10 +02:00
|
|
|
};
|
|
|
|
in "${combinedExtensionsDrv}/${subDir}";
|
2022-02-07 18:40:59 +01:00
|
|
|
}))
|
2024-07-16 14:09:45 +02:00
|
|
|
]);
|
2018-11-19 17:50:35 +01:00
|
|
|
};
|
|
|
|
}
|