mirror of
https://github.com/nix-community/nixvim.git
synced 2025-06-20 16:15:43 +02:00
137 lines
4.4 KiB
Nix
137 lines
4.4 KiB
Nix
{ lib }:
|
||
rec {
|
||
# Get a (sub)option by walking the path,
|
||
# checking for submodules along the way
|
||
getOptionRecursive =
|
||
opt: prefix: optionPath:
|
||
if optionPath == [ ] then
|
||
opt
|
||
else if lib.isOption opt then
|
||
getOptionRecursive (opt.type.getSubOptions prefix) prefix optionPath
|
||
else
|
||
let
|
||
name = lib.head optionPath;
|
||
opt' = lib.getAttr name opt;
|
||
prefix' = prefix ++ [ name ];
|
||
optionPath' = lib.drop 1 optionPath;
|
||
in
|
||
getOptionRecursive opt' prefix' optionPath';
|
||
|
||
# Like mkRemovedOptionModule, but has support for nested sub-options
|
||
# and uses warnings instead of assertions.
|
||
mkDeprecatedSubOptionModule =
|
||
optionPath: replacementInstructions:
|
||
{ options, ... }:
|
||
{
|
||
options = lib.setAttrByPath optionPath (
|
||
lib.mkOption {
|
||
# When (e.g.) `mkAttrs` is used on a submodule, this option will be evaluated.
|
||
# Therefore we have to apply _something_ (null) when there's no definition.
|
||
apply =
|
||
v:
|
||
let
|
||
# Avoid "option used but not defined" errors
|
||
res = builtins.tryEval v;
|
||
in
|
||
if res.success then res.value else null;
|
||
visible = false;
|
||
}
|
||
);
|
||
config.warnings =
|
||
let
|
||
opt = getOptionRecursive options [ ] optionPath;
|
||
in
|
||
lib.optional opt.isDefined ''
|
||
The option definition `${lib.showOption optionPath}' in ${lib.showFiles opt.files} is deprecated.
|
||
${replacementInstructions}
|
||
'';
|
||
};
|
||
|
||
/*
|
||
Returns a function that maps
|
||
[
|
||
"someOption"
|
||
["fooBar" "someSubOption"]
|
||
{ old = "someOtherOption"; new = ["foo_bar" "some_other_option"]}
|
||
]
|
||
|
||
to
|
||
[
|
||
(lib.mkRenamedOptionModule
|
||
(oldPath ++ ["someOption"])
|
||
(newPath ++ ["some_option"])
|
||
)
|
||
(lib.mkRenamedOptionModule
|
||
(oldPath ++ ["fooBar" "someSubOption"])
|
||
(newPath ++ ["foo_bar" "some_sub_option"])
|
||
)
|
||
(lib.mkRenamedOptionModule
|
||
(oldPath ++ ["someOtherOption"])
|
||
(newPath ++ ["foo_bar" "some_other_option"])
|
||
)
|
||
]
|
||
*/
|
||
mkSettingsRenamedOptionModules =
|
||
oldPrefix: newPrefix:
|
||
map (
|
||
spec:
|
||
let
|
||
finalSpec =
|
||
if lib.isAttrs spec then
|
||
lib.mapAttrs (_: lib.toList) spec
|
||
else
|
||
{
|
||
old = lib.toList spec;
|
||
new = map lib.nixvim.toSnakeCase finalSpec.old;
|
||
};
|
||
in
|
||
lib.mkRenamedOptionModule (oldPrefix ++ finalSpec.old) (newPrefix ++ finalSpec.new)
|
||
);
|
||
|
||
# A clone of types.coercedTo, but it prints a warning when oldType is used.
|
||
transitionType =
|
||
oldType: coerceFn: newType:
|
||
assert lib.assertMsg (
|
||
oldType.getSubModules == null
|
||
) "transitionType: oldType must not have submodules (it’s a ${oldType.description})";
|
||
lib.mkOptionType rec {
|
||
name = "transitionType";
|
||
inherit (newType) description;
|
||
check = x: (oldType.check x && newType.check (coerceFn x)) || newType.check x;
|
||
merge =
|
||
opt: defs:
|
||
let
|
||
coerceVal =
|
||
val:
|
||
if oldType.check val then
|
||
lib.warn ''
|
||
Passing a ${oldType.description} for `${lib.showOption opt}' is deprecated, use ${newType.description} instead. Definitions: ${lib.options.showDefs defs}
|
||
'' (coerceFn val)
|
||
else
|
||
val;
|
||
in
|
||
newType.merge opt (map (def: def // { value = coerceVal def.value; }) defs);
|
||
inherit (newType) emptyValue;
|
||
inherit (newType) getSubOptions;
|
||
inherit (newType) getSubModules;
|
||
substSubModules = m: transitionType oldType coerceFn (newType.substSubModules m);
|
||
typeMerge = t1: t2: null;
|
||
functor = (lib.types.defaultFunctor name) // {
|
||
wrapped = newType;
|
||
};
|
||
nestedTypes.coercedType = oldType;
|
||
nestedTypes.finalType = newType;
|
||
};
|
||
|
||
mkRemovedPackageOptionModule =
|
||
{
|
||
plugin,
|
||
packageName,
|
||
oldPackageName ? packageName,
|
||
}:
|
||
lib.mkRemovedOptionModule [ "plugins" plugin "${oldPackageName}Package" ] ''
|
||
Please use the `dependencies.${packageName}` top-level option instead:
|
||
- `dependencies.${packageName}.enable = false` to disable installing `${packageName}`
|
||
- `dependencies.${packageName}.package` to choose which package to install for `${packageName}`.
|
||
'';
|
||
}
|