mirror of
https://github.com/sindresorhus/type-fest.git
synced 2025-12-08 19:25:05 +00:00
155 lines
5.4 KiB
TypeScript
155 lines
5.4 KiB
TypeScript
import type {ApplyDefaultOptions, BuiltIns} from './internal/index.d.ts';
|
|
import type {IsNever} from './is-never.d.ts';
|
|
|
|
/**
|
|
@see {@link PartialDeep}
|
|
*/
|
|
export type PartialDeepOptions = {
|
|
/**
|
|
Whether to affect the individual elements of arrays and tuples.
|
|
|
|
@default false
|
|
*/
|
|
readonly recurseIntoArrays?: boolean;
|
|
|
|
/**
|
|
Allows `undefined` values in non-tuple arrays.
|
|
|
|
- When set to `true`, elements of non-tuple arrays can be `undefined`.
|
|
- When set to `false`, only explicitly defined elements are allowed in non-tuple arrays, ensuring stricter type checking.
|
|
|
|
@default false
|
|
|
|
@example
|
|
You can allow `undefined` values in non-tuple arrays by passing `{recurseIntoArrays: true; allowUndefinedInNonTupleArrays: true}` as the second type argument:
|
|
|
|
```
|
|
import type {PartialDeep} from 'type-fest';
|
|
|
|
type Settings = {
|
|
languages: string[];
|
|
};
|
|
|
|
declare const partialSettings: PartialDeep<Settings, {recurseIntoArrays: true; allowUndefinedInNonTupleArrays: true}>;
|
|
|
|
partialSettings.languages = [undefined]; // OK
|
|
```
|
|
*/
|
|
readonly allowUndefinedInNonTupleArrays?: boolean;
|
|
};
|
|
|
|
type DefaultPartialDeepOptions = {
|
|
recurseIntoArrays: false;
|
|
allowUndefinedInNonTupleArrays: false;
|
|
};
|
|
|
|
/**
|
|
Create a type from another type with all keys and nested keys set to optional.
|
|
|
|
Use-cases:
|
|
- Merging a default settings/config object with another object, the second object would be a deep partial of the default object.
|
|
- Mocking and testing complex entities, where populating an entire object with its keys would be redundant in terms of the mock or test.
|
|
|
|
@example
|
|
```
|
|
import type {PartialDeep} from 'type-fest';
|
|
|
|
let settings = {
|
|
textEditor: {
|
|
fontSize: 14,
|
|
fontColor: '#000000',
|
|
fontWeight: 400,
|
|
},
|
|
autocomplete: false,
|
|
autosave: true,
|
|
};
|
|
|
|
const applySavedSettings = (savedSettings: PartialDeep<typeof settings>) => (
|
|
{...settings, ...savedSettings, textEditor: {...settings.textEditor, ...savedSettings.textEditor}}
|
|
);
|
|
|
|
settings = applySavedSettings({textEditor: {fontWeight: 500}});
|
|
```
|
|
|
|
By default, this does not affect elements in array and tuple types. You can change this by passing `{recurseIntoArrays: true}` as the second type argument:
|
|
|
|
```
|
|
import type {PartialDeep} from 'type-fest';
|
|
|
|
type Shape = {
|
|
dimensions: [number, number];
|
|
};
|
|
|
|
const partialShape: PartialDeep<Shape, {recurseIntoArrays: true}> = {
|
|
dimensions: [], // OK
|
|
};
|
|
|
|
partialShape.dimensions = [15]; // OK
|
|
```
|
|
|
|
@see {@link PartialDeepOptions}
|
|
|
|
@category Object
|
|
@category Array
|
|
@category Set
|
|
@category Map
|
|
*/
|
|
export type PartialDeep<T, Options extends PartialDeepOptions = {}> =
|
|
_PartialDeep<T, ApplyDefaultOptions<PartialDeepOptions, DefaultPartialDeepOptions, Options>>;
|
|
|
|
type _PartialDeep<T, Options extends Required<PartialDeepOptions>> = T extends BuiltIns | ((new (...arguments_: any[]) => unknown))
|
|
? T
|
|
: IsNever<keyof T> extends true // For functions with no properties
|
|
? T
|
|
: T extends Map<infer KeyType, infer ValueType>
|
|
? PartialMapDeep<KeyType, ValueType, Options>
|
|
: T extends Set<infer ItemType>
|
|
? PartialSetDeep<ItemType, Options>
|
|
: T extends ReadonlyMap<infer KeyType, infer ValueType>
|
|
? PartialReadonlyMapDeep<KeyType, ValueType, Options>
|
|
: T extends ReadonlySet<infer ItemType>
|
|
? PartialReadonlySetDeep<ItemType, Options>
|
|
: T extends object
|
|
? T extends ReadonlyArray<infer ItemType> // Test for arrays/tuples, per https://github.com/microsoft/TypeScript/issues/35156
|
|
? Options['recurseIntoArrays'] extends true
|
|
? ItemType[] extends T // Test for arrays (non-tuples) specifically
|
|
? readonly ItemType[] extends T // Differentiate readonly and mutable arrays
|
|
? ReadonlyArray<_PartialDeep<Options['allowUndefinedInNonTupleArrays'] extends false ? ItemType : ItemType | undefined, Options>>
|
|
: Array<_PartialDeep<Options['allowUndefinedInNonTupleArrays'] extends false ? ItemType : ItemType | undefined, Options>>
|
|
: PartialObjectDeep<T, Options> // Tuples behave properly
|
|
: T // If they don't opt into array testing, just use the original type
|
|
: PartialObjectDeep<T, Options>
|
|
: unknown;
|
|
|
|
/**
|
|
Same as `PartialDeep`, but accepts only `Map`s and as inputs. Internal helper for `PartialDeep`.
|
|
*/
|
|
type PartialMapDeep<KeyType, ValueType, Options extends Required<PartialDeepOptions>> = {} & Map<_PartialDeep<KeyType, Options>, _PartialDeep<ValueType, Options>>;
|
|
|
|
/**
|
|
Same as `PartialDeep`, but accepts only `Set`s as inputs. Internal helper for `PartialDeep`.
|
|
*/
|
|
type PartialSetDeep<T, Options extends Required<PartialDeepOptions>> = {} & Set<_PartialDeep<T, Options>>;
|
|
|
|
/**
|
|
Same as `PartialDeep`, but accepts only `ReadonlyMap`s as inputs. Internal helper for `PartialDeep`.
|
|
*/
|
|
type PartialReadonlyMapDeep<KeyType, ValueType, Options extends Required<PartialDeepOptions>> = {} & ReadonlyMap<_PartialDeep<KeyType, Options>, _PartialDeep<ValueType, Options>>;
|
|
|
|
/**
|
|
Same as `PartialDeep`, but accepts only `ReadonlySet`s as inputs. Internal helper for `PartialDeep`.
|
|
*/
|
|
type PartialReadonlySetDeep<T, Options extends Required<PartialDeepOptions>> = {} & ReadonlySet<_PartialDeep<T, Options>>;
|
|
|
|
/**
|
|
Same as `PartialDeep`, but accepts only `object`s as inputs. Internal helper for `PartialDeep`.
|
|
*/
|
|
type PartialObjectDeep<ObjectType extends object, Options extends Required<PartialDeepOptions>> =
|
|
(ObjectType extends (...arguments_: any) => unknown
|
|
? (...arguments_: Parameters<ObjectType>) => ReturnType<ObjectType>
|
|
: {}) & ({
|
|
[KeyType in keyof ObjectType]?: _PartialDeep<ObjectType[KeyType], Options>
|
|
});
|
|
|
|
export {};
|