Skip to content

Commit beaabe1

Browse files
authoredAug 24, 2022
Add ConditionalSimplify and ConditionalSimplifyDeep types (#442)
1 parent bbccfb8 commit beaabe1

File tree

2 files changed

+113
-0
lines changed

2 files changed

+113
-0
lines changed
 

‎source/conditional-simplify.d.ts

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
/**
2+
Simplifies a type while including and/or excluding certain types from being simplified. Useful to improve type hints shown in editors. And also to transform an interface into a type to aide with assignability.
3+
4+
This type is **experimental** and was introduced as a result of this {@link https://github.com/sindresorhus/type-fest/issues/436 issue}. It should be used with caution.
5+
6+
@internal
7+
@experimental
8+
@see Simplify
9+
@category Object
10+
*/
11+
export type ConditionalSimplify<Type, ExcludeType = never, IncludeType = unknown> = Type extends ExcludeType
12+
? Type
13+
: Type extends IncludeType
14+
? {[TypeKey in keyof Type]: Type[TypeKey]}
15+
: Type;
16+
17+
/**
18+
Recursively simplifies a type while including and/or excluding certain types from being simplified.
19+
20+
This type is **experimental** and was introduced as a result of this {@link https://github.com/sindresorhus/type-fest/issues/436 issue}. It should be used with caution.
21+
22+
See {@link ConditionalSimplify} for usages and examples.
23+
24+
@internal
25+
@experimental
26+
@category Object
27+
*/
28+
export type ConditionalSimplifyDeep<Type, ExcludeType = never, IncludeType = unknown> = Type extends ExcludeType
29+
? Type
30+
: Type extends IncludeType
31+
? {[TypeKey in keyof Type]: ConditionalSimplifyDeep<Type[TypeKey], ExcludeType, IncludeType>}
32+
: Type;

‎test-d/conditional-simplify.ts

+81
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
import {expectError, expectType} from 'tsd';
2+
import type {ConditionalSimplify, ConditionalSimplifyDeep} from '../source/conditional-simplify';
3+
4+
type Position = {top: number; left: number};
5+
type Size = {width: number; height: number};
6+
7+
// In your editor, hovering over `PositionAndSizeSimplified` will show a simplified object with all the properties.
8+
type PositionAndSizeIntersection = Position & Size;
9+
type PositionAndSizeSimplified = ConditionalSimplify<PositionAndSizeIntersection>;
10+
11+
const position = {top: 120, left: 240};
12+
const size = {width: 480, height: 600};
13+
const positionAndSize = {...position, ...size};
14+
expectType<PositionAndSizeSimplified>(positionAndSize);
15+
16+
// Exclude function type to be simplified.
17+
type SomeFunction = (type: string) => string;
18+
type SimplifiedFunctionFail = ConditionalSimplify<SomeFunction>; // Return '{}'
19+
type SimplifiedFunctionPass = ConditionalSimplify<SomeFunction, Function>; // Return '(type: string) => string'
20+
21+
declare const simplifiedFunctionFail: SimplifiedFunctionFail;
22+
declare const simplifiedFunctionPass: SimplifiedFunctionPass;
23+
24+
expectError<SomeFunction>(simplifiedFunctionFail);
25+
expectType<SomeFunction>(simplifiedFunctionPass);
26+
27+
// Should simplify interface deeply.
28+
interface SomeNode {
29+
parent: PositionAndSizeIntersection;
30+
childs: Array<{parent: PositionAndSizeIntersection}>;
31+
}
32+
33+
// In your editor, hovering over `SomeNodeSimplified` will show a simplified object with all the properties.
34+
type SomeNodeSimplified = ConditionalSimplifyDeep<SomeNode>;
35+
36+
const someNode = {parent: positionAndSize, childs: [{parent: positionAndSize}, {parent: positionAndSize}]};
37+
expectType<SomeNodeSimplified>(someNode);
38+
39+
// Should simplify interface deeply excluding Function type.
40+
interface MovablePosition extends Position {
41+
move(position: Position): Position;
42+
}
43+
44+
interface MovableCollection {
45+
position: MovablePosition;
46+
top: {position: MovablePosition; size: Size};
47+
left: {position: MovablePosition; size: Size};
48+
}
49+
50+
type MovableNodeSimplifiedFail = ConditionalSimplifyDeep<MovableCollection>;
51+
type MovableNodeSimplifiedPass = ConditionalSimplifyDeep<MovableCollection, Function>;
52+
53+
declare const movableNodeSimplifiedFail: MovableNodeSimplifiedFail;
54+
declare const movableNodeSimplifiedPass: MovableNodeSimplifiedPass;
55+
56+
expectError<MovableCollection>(movableNodeSimplifiedFail);
57+
expectType<MovableCollection>(movableNodeSimplifiedPass);
58+
59+
const movablePosition = {
60+
top: 42,
61+
left: 42,
62+
move(position: Position) {
63+
return position;
64+
},
65+
};
66+
67+
const movableNode = {
68+
position: movablePosition,
69+
top: {position: movablePosition, size},
70+
left: {position: movablePosition, size},
71+
};
72+
73+
expectType<MovableNodeSimplifiedPass>(movableNode);
74+
75+
// Should exclude `Function` and `Size` type (mainly visual, mouse over the statement).
76+
type ExcludeFunctionAndSize1 = ConditionalSimplifyDeep<MovableCollection, Function | Size>;
77+
expectType<ExcludeFunctionAndSize1>(movableNode);
78+
79+
// Same as above but using `IncludeType` parameter (mainly visual, mouse over the statement).
80+
type ExcludeFunctionAndSize2 = ConditionalSimplifyDeep<MovableCollection, Function, MovableCollection | Position>;
81+
expectType<ExcludeFunctionAndSize2>(movableNode);

0 commit comments

Comments
 (0)
Please sign in to comment.