1
0
Fork 0
mirror of https://github.com/nix-community/home-manager synced 2025-01-23 09:29:50 +01:00

vscode: add initial profiles support

Signed-off-by: Reputable2722 <153411261+Reputable2772@users.noreply.github.com>
This commit is contained in:
Reputable2722 2024-07-16 17:39:45 +05:30
parent 90ae324e2c
commit 6266e351d4
No known key found for this signature in database

View file

@ -1,5 +1,7 @@
{ config, lib, pkgs, ... }: { config, lib, pkgs, ... }:
# TODO: Re-write tests to support profiles.
with lib; with lib;
let let
@ -30,28 +32,174 @@ let
else else
"${config.xdg.configHome}/${configDir}/User"; "${config.xdg.configHome}/${configDir}/User";
configFilePath = "${userDir}/settings.json"; configFilePath = name:
tasksFilePath = "${userDir}/tasks.json"; "${userDir}/${
keybindingsFilePath = "${userDir}/keybindings.json"; 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 = "${userDir}/snippets"; snippetDir = name:
"${userDir}/${
optionalString (name != "default") "profiles/${name}/"
}snippets";
# TODO: On Darwin where are the extensions? # TODO: On Darwin where are the extensions?
extensionPath = ".${extensionDir}/extensions"; extensionPath = ".${extensionDir}/extensions";
extensionJson = pkgs.vscode-utils.toExtensionJson cfg.extensions; extensionJson = ext: pkgs.vscode-utils.toExtensionJson ext;
extensionJsonFile = pkgs.writeTextFile { extensionJsonFile = name: text:
name = "extensions-json"; pkgs.writeTextFile {
destination = "/share/vscode/extensions/extensions.json"; inherit text;
text = extensionJson; name = "extensions-json-${name}";
}; destination = "/share/vscode/extensions/extensions.json";
};
mergedUserSettings = cfg.userSettings mergedUserSettings = userSettings:
userSettings
// optionalAttrs (!cfg.enableUpdateCheck) { "update.mode" = "none"; } // optionalAttrs (!cfg.enableUpdateCheck) { "update.mode" = "none"; }
// optionalAttrs (!cfg.enableExtensionUpdateCheck) { // optionalAttrs (!cfg.enableExtensionUpdateCheck) {
"extensions.autoCheckUpdates" = false; "extensions.autoCheckUpdates" = false;
}; };
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`.
'';
};
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`.
'';
};
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 ''
[
{
key = "ctrl+c";
command = "editor.action.clipboardCopyAction";
when = "textInputFocus";
}
]
'';
description = ''
Keybindings written to Visual Studio Code's
{file}`keybindings.json`.
'';
};
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.
'';
};
languageSnippets = mkOption {
type = jsonFormat.type;
default = { };
example = {
haskell = {
fixme = {
prefix = [ "fixme" ];
body = [ "$LINE_COMMENT FIXME: $0" ];
description = "Insert a FIXME remark";
};
};
};
description = "Defines user snippets for different languages.";
};
globalSnippets = mkOption {
type = jsonFormat.type;
default = { };
example = {
fixme = {
prefix = [ "fixme" ];
body = [ "$LINE_COMMENT FIXME: $0" ];
description = "Insert a FIXME remark";
};
};
description = "Defines global user snippets.";
};
} // optionalAttrs default {
name = mkOption {
type = types.str;
description = "Visual Studio Code's Profile name.";
};
};
};
allProfiles = cfg.profiles ++ [ cfg.defaultProfile ];
in { in {
# TODO: Backwards compatibility with old options.
imports = [ imports = [
(mkChangedOptionModule [ "programs" "vscode" "immutableExtensionsDir" ] [ (mkChangedOptionModule [ "programs" "vscode" "immutableExtensionsDir" ] [
"programs" "programs"
@ -60,204 +208,134 @@ in {
] (config: !config.programs.vscode.immutableExtensionsDir)) ] (config: !config.programs.vscode.immutableExtensionsDir))
]; ];
options = { options.programs.vscode = {
programs.vscode = { enable = mkEnableOption "Visual Studio Code";
enable = mkEnableOption "Visual Studio Code";
package = mkOption { package = mkOption {
type = types.package; type = types.package;
default = pkgs.vscode; default = pkgs.vscode;
defaultText = literalExpression "pkgs.vscode"; defaultText = literalExpression "pkgs.vscode";
example = literalExpression "pkgs.vscodium"; example = literalExpression "pkgs.vscodium";
description = '' description = ''
Version of Visual Studio Code to install. Version of Visual Studio Code to install.
''; '';
}; };
enableUpdateCheck = mkOption { enableUpdateCheck = mkOption {
type = types.bool; type = types.bool;
default = true; default = true;
description = '' description = ''
Whether to enable update checks/notifications. Whether to enable update checks/notifications.
''; '';
}; };
enableExtensionUpdateCheck = mkOption { enableExtensionUpdateCheck = mkOption {
type = types.bool; type = types.bool;
default = true; default = true;
description = '' description = ''
Whether to enable update notifications for extensions. Whether to enable update notifications for extensions.
''; '';
}; };
userSettings = mkOption { mutableExtensionsDir = mkOption {
type = jsonFormat.type; type = types.bool;
default = { }; default = cfg.profiles == [ ];
example = literalExpression '' example = false;
{ description = ''
"files.autoSave" = "off"; Whether extensions can be installed or updated manually
"[nix]"."editor.tabSize" = 2; or by Visual Studio Code. This option is effective only
} when there is a single profile (i.e. default).
''; '';
description = '' };
Configuration written to Visual Studio Code's
{file}`settings.json`.
'';
};
userTasks = mkOption { profiles = mkOption {
type = jsonFormat.type; type = types.listOf (profileType true);
default = { }; default = [ ];
example = literalExpression '' };
{ defaultProfile = mkOption {
version = "2.0.0"; type = profileType false;
tasks = [ default = { };
{
type = "shell";
label = "Hello task";
command = "hello";
}
];
}
'';
description = ''
Configuration written to Visual Studio Code's
{file}`tasks.json`.
'';
};
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 ''
[
{
key = "ctrl+c";
command = "editor.action.clipboardCopyAction";
when = "textInputFocus";
}
]
'';
description = ''
Keybindings written to Visual Studio Code's
{file}`keybindings.json`.
'';
};
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.
'';
};
mutableExtensionsDir = mkOption {
type = types.bool;
default = true;
example = false;
description = ''
Whether extensions can be installed or updated manually
or by Visual Studio Code.
'';
};
languageSnippets = mkOption {
type = jsonFormat.type;
default = { };
example = {
haskell = {
fixme = {
prefix = [ "fixme" ];
body = [ "$LINE_COMMENT FIXME: $0" ];
description = "Insert a FIXME remark";
};
};
};
description = "Defines user snippets for different languages.";
};
globalSnippets = mkOption {
type = jsonFormat.type;
default = { };
example = {
fixme = {
prefix = [ "fixme" ];
body = [ "$LINE_COMMENT FIXME: $0" ];
description = "Insert a FIXME remark";
};
};
description = "Defines global user snippets.";
};
}; };
}; };
config = mkIf cfg.enable { config = mkIf cfg.enable {
warnings = [
(mkIf (cfg.profiles != [ ] && cfg.mutableExtensionsDir)
"programs.vscode.mutableExtensionsDir can be used only if profiles is an empty list.")
];
home.packages = [ cfg.package ]; home.packages = [ cfg.package ];
home.file = mkMerge [ /* *
(mkIf (mergedUserSettings != { }) { TODO: Write a home.activation script for ${userDir}/globalStorage/storage.json, appending
"${configFilePath}".source = every profile in the format `{ "name": <profile_name>, "location": <profile_name> }` to the
jsonFormat.generate "vscode-user-settings" mergedUserSettings; userDataProfiles array.
})
(mkIf (cfg.userTasks != { }) {
"${tasksFilePath}".source =
jsonFormat.generate "vscode-user-tasks" cfg.userTasks;
})
(mkIf (cfg.keybindings != [ ])
(let dropNullFields = filterAttrs (_: v: v != null);
in {
"${keybindingsFilePath}".source =
jsonFormat.generate "vscode-keybindings"
(map dropNullFields cfg.keybindings);
}))
(mkIf (cfg.extensions != [ ]) (let
subDir = "share/vscode/extensions";
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);
})
(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
# Adapted from https://discourse.nixos.org/t/vscode-extensions-setup/1801/2 # Adapted from https://discourse.nixos.org/t/vscode-extensions-setup/1801/2
subDir = "share/vscode/extensions";
toPaths = ext: toPaths = ext:
map (k: { "${extensionPath}/${k}".source = "${ext}/${subDir}/${k}"; }) map (k: { "${extensionPath}/${k}".source = "${ext}/${subDir}/${k}"; })
(if ext ? vscodeExtUniqueId then (if ext ? vscodeExtUniqueId then
[ ext.vscodeExtUniqueId ] [ ext.vscodeExtUniqueId ]
else else
builtins.attrNames (builtins.readDir (ext + "/${subDir}"))); builtins.attrNames (builtins.readDir (ext + "/${subDir}")));
in if cfg.mutableExtensionsDir then in if (cfg.mutableExtensionsDir && cfg.profiles == [ ]) then
mkMerge (concatMap toPaths cfg.extensions mkMerge (concatMap toPaths (flatten (map (v: v.extensions) allProfiles))
++ lib.optional (lib.versionAtLeast vscodeVersion "1.74.0") { ++ lib.optional (lib.versionAtLeast vscodeVersion "1.74.0") {
# Whenever our immutable extensions.json changes, force VSCode to regenerate # Whenever our immutable extensions.json changes, force VSCode to regenerate
# extensions.json with both mutable and immutable extensions. # extensions.json with both mutable and immutable extensions.
"${extensionPath}/.extensions-immutable.json" = { "${extensionPath}/.extensions-immutable.json" = {
text = extensionJson; text = extensionJson cfg.defaultProfile.extensions;
onChange = '' onChange = ''
run rm $VERBOSE_ARG -f ${extensionPath}/{extensions.json,.init-default-profile-extensions} run rm $VERBOSE_ARG -f ${extensionPath}/{extensions.json,.init-default-profile-extensions}
verboseEcho "Regenerating VSCode extensions.json" verboseEcho "Regenerating VSCode extensions.json"
@ -269,25 +347,13 @@ in {
"${extensionPath}".source = let "${extensionPath}".source = let
combinedExtensionsDrv = pkgs.buildEnv { combinedExtensionsDrv = pkgs.buildEnv {
name = "vscode-extensions"; name = "vscode-extensions";
paths = cfg.extensions paths = flatten (map (v: v.extensions) allProfiles) ++ lib.optional
++ lib.optional (lib.versionAtLeast vscodeVersion "1.74.0") (lib.versionAtLeast vscodeVersion "1.74.0" && cfg.defaultProfile
extensionJsonFile; != { }) (extensionJsonFile "default"
(extensionJson cfg.defaultProfile.extensions));
}; };
in "${combinedExtensionsDrv}/${subDir}"; in "${combinedExtensionsDrv}/${subDir}";
})) }))
]);
(mkIf (cfg.globalSnippets != { })
(let globalSnippets = "${snippetDir}/global.code-snippets";
in {
"${globalSnippets}".source =
jsonFormat.generate "user-snippet-global.code-snippets"
cfg.globalSnippets;
}))
(lib.mapAttrs' (language: snippet:
lib.nameValuePair "${snippetDir}/${language}.json" {
source = jsonFormat.generate "user-snippet-${language}.json" snippet;
}) cfg.languageSnippets)
];
}; };
} }