mirror of
https://github.com/sindresorhus/type-fest.git
synced 2025-12-08 19:25:05 +00:00
48 lines
1.3 KiB
TypeScript
48 lines
1.3 KiB
TypeScript
import type {IfNever} from './if-never.d.ts';
|
|
|
|
/**
|
|
Extract the keys from a type where the value type of the key extends the given `Condition`.
|
|
|
|
Internally this is used for the `ConditionalPick` and `ConditionalExcept` types.
|
|
|
|
@example
|
|
```
|
|
import type {ConditionalKeys} from 'type-fest';
|
|
|
|
interface Example {
|
|
a: string;
|
|
b: string | number;
|
|
c?: string;
|
|
d: {};
|
|
}
|
|
|
|
type StringKeysOnly = ConditionalKeys<Example, string>;
|
|
//=> 'a'
|
|
```
|
|
|
|
To support partial types, make sure your `Condition` is a union of undefined (for example, `string | undefined`) as demonstrated below.
|
|
|
|
@example
|
|
```
|
|
import type {ConditionalKeys} from 'type-fest';
|
|
|
|
type StringKeysAndUndefined = ConditionalKeys<Example, string | undefined>;
|
|
//=> 'a' | 'c'
|
|
```
|
|
|
|
@category Object
|
|
*/
|
|
export type ConditionalKeys<Base, Condition> =
|
|
{
|
|
// Map through all the keys of the given base type.
|
|
[Key in keyof Base]-?:
|
|
// Pick only keys with types extending the given `Condition` type.
|
|
Base[Key] extends Condition
|
|
// Retain this key
|
|
// If the value for the key extends never, only include it if `Condition` also extends never
|
|
? IfNever<Base[Key], IfNever<Condition, Key, never>, Key>
|
|
// Discard this key since the condition fails.
|
|
: never;
|
|
// Convert the produced object into a union type of the keys which passed the conditional test.
|
|
}[keyof Base];
|