Skip to content

Commit 1974944

Browse files
andrewborsteinsom-smsindresorhus
authored
DelimiterCase: Pass Options generic to all related types (#1078)
Co-authored-by: Som Shekhar Mukherjee <[email protected]> Co-authored-by: Sindre Sorhus <[email protected]>
1 parent c11c9ca commit 1974944

13 files changed

+143
-37
lines changed

source/delimiter-cased-properties-deep.d.ts

+42-18
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
1-
import type {DelimiterCase} from './delimiter-case';
2-
import type {NonRecursiveType} from './internal';
1+
import type {DefaultDelimiterCaseOptions, DelimiterCase} from './delimiter-case';
2+
import type {ApplyDefaultOptions, NonRecursiveType} from './internal';
33
import type {UnknownArray} from './unknown-array';
4+
import type {WordsOptions} from './words';
45

56
/**
67
Convert object properties to delimiter case recursively.
@@ -40,6 +41,16 @@ const result: DelimiterCasedPropertiesDeep<UserWithFriends, '-'> = {
4041
},
4142
],
4243
};
44+
45+
const splitOnNumbers: DelimiterCasedPropertiesDeep<{line1: { line2: [{ line3: string }] }}, '-', {splitOnNumbers: true}> = {
46+
'line-1': {
47+
'line-2': [
48+
{
49+
'line-3': 'string',
50+
},
51+
],
52+
},
53+
};
4354
```
4455
4556
@category Change case
@@ -49,34 +60,47 @@ const result: DelimiterCasedPropertiesDeep<UserWithFriends, '-'> = {
4960
export type DelimiterCasedPropertiesDeep<
5061
Value,
5162
Delimiter extends string,
63+
Options extends WordsOptions = {},
64+
> = _DelimiterCasedPropertiesDeep<Value, Delimiter, ApplyDefaultOptions<WordsOptions, DefaultDelimiterCaseOptions, Options>>;
65+
66+
type _DelimiterCasedPropertiesDeep<
67+
Value,
68+
Delimiter extends string,
69+
Options extends Required<WordsOptions>,
5270
> = Value extends NonRecursiveType
5371
? Value
5472
: Value extends UnknownArray
55-
? DelimiterCasedPropertiesArrayDeep<Value, Delimiter>
73+
? DelimiterCasedPropertiesArrayDeep<Value, Delimiter, Options>
5674
: Value extends Set<infer U>
57-
? Set<DelimiterCasedPropertiesDeep<U, Delimiter>> : {
75+
? Set<_DelimiterCasedPropertiesDeep<U, Delimiter, Options>> : {
5876
[K in keyof Value as DelimiterCase<
5977
K,
60-
Delimiter
61-
>]: DelimiterCasedPropertiesDeep<Value[K], Delimiter>;
78+
Delimiter,
79+
Options
80+
>]: _DelimiterCasedPropertiesDeep<Value[K], Delimiter, Options>;
6281
};
6382

6483
// This is a copy of CamelCasedPropertiesArrayDeep (see: camel-cased-properties-deep.d.ts).
6584
// These types should be kept in sync.
66-
type DelimiterCasedPropertiesArrayDeep<Value extends UnknownArray, Delimiter extends string> =
67-
Value extends []
68-
? []
69-
// Tailing spread array
70-
: Value extends [infer U, ...infer V]
71-
? [DelimiterCasedPropertiesDeep<U, Delimiter>, ...DelimiterCasedPropertiesDeep<V, Delimiter>]
72-
: Value extends readonly [infer U, ...infer V]
73-
? readonly [DelimiterCasedPropertiesDeep<U, Delimiter>, ...DelimiterCasedPropertiesDeep<V, Delimiter>]
74-
// Leading spread array
85+
type DelimiterCasedPropertiesArrayDeep<
86+
Value extends UnknownArray,
87+
Delimiter extends string,
88+
Options extends Required<WordsOptions>,
89+
> = Value extends []
90+
? []
91+
// Trailing spread array
92+
: Value extends [infer U, ...infer V]
93+
? [_DelimiterCasedPropertiesDeep<U, Delimiter, Options>, ..._DelimiterCasedPropertiesDeep<V, Delimiter, Options>]
94+
: Value extends readonly [infer U, ...infer V]
95+
? readonly [_DelimiterCasedPropertiesDeep<U, Delimiter, Options>, ..._DelimiterCasedPropertiesDeep<V, Delimiter, Options>]
96+
// Leading spread array
97+
: Value extends [...infer U, infer V]
98+
? [..._DelimiterCasedPropertiesDeep<U, Delimiter, Options>, _DelimiterCasedPropertiesDeep<V, Delimiter, Options>]
7599
: Value extends readonly [...infer U, infer V]
76-
? [...DelimiterCasedPropertiesDeep<U, Delimiter>, DelimiterCasedPropertiesDeep<V, Delimiter>]
100+
? readonly [..._DelimiterCasedPropertiesDeep<U, Delimiter, Options>, _DelimiterCasedPropertiesDeep<V, Delimiter, Options>]
77101
// Array
78102
: Value extends Array<infer U>
79-
? Array<DelimiterCasedPropertiesDeep<U, Delimiter>>
103+
? Array<_DelimiterCasedPropertiesDeep<U, Delimiter, Options>>
80104
: Value extends ReadonlyArray<infer U>
81-
? ReadonlyArray<DelimiterCasedPropertiesDeep<U, Delimiter>>
105+
? ReadonlyArray<_DelimiterCasedPropertiesDeep<U, Delimiter, Options>>
82106
: never;

source/delimiter-cased-properties.d.ts

+11-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
1-
import type {DelimiterCase} from './delimiter-case';
1+
import type {DefaultDelimiterCaseOptions, DelimiterCase} from './delimiter-case';
2+
import type {ApplyDefaultOptions} from './internal';
3+
import type {WordsOptions} from './words';
24

35
/**
46
Convert object properties to delimiter case but not recursively.
@@ -21,6 +23,10 @@ const result: DelimiterCasedProperties<User, '-'> = {
2123
'user-id': 1,
2224
'user-name': 'Tom',
2325
};
26+
27+
const splitOnNumbers: DelimiterCasedProperties<{line1: string}, '-', {splitOnNumbers: true}> = {
28+
'line-1': 'string',
29+
};
2430
```
2531
2632
@category Change case
@@ -30,8 +36,11 @@ const result: DelimiterCasedProperties<User, '-'> = {
3036
export type DelimiterCasedProperties<
3137
Value,
3238
Delimiter extends string,
39+
Options extends WordsOptions = {},
3340
> = Value extends Function
3441
? Value
3542
: Value extends Array<infer U>
3643
? Value
37-
: {[K in keyof Value as DelimiterCase<K, Delimiter>]: Value[K]};
44+
: {[K in keyof Value as
45+
DelimiterCase<K, Delimiter, ApplyDefaultOptions<WordsOptions, DefaultDelimiterCaseOptions, Options>>
46+
]: Value[K]};

source/kebab-cased-properties-deep.d.ts

+17-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,7 @@
1+
import type {DefaultDelimiterCaseOptions} from './delimiter-case';
12
import type {DelimiterCasedPropertiesDeep} from './delimiter-cased-properties-deep';
3+
import type {ApplyDefaultOptions} from './internal';
4+
import type {WordsOptions} from './words';
25

36
/**
47
Convert object properties to kebab case recursively.
@@ -38,10 +41,23 @@ const result: KebabCasedPropertiesDeep<UserWithFriends> = {
3841
},
3942
],
4043
};
44+
45+
const splitOnNumbers: KebabCasedPropertiesDeep<{line1: { line2: [{ line3: string }] }}, {splitOnNumbers: true}> = {
46+
'line-1': {
47+
'line-2': [
48+
{
49+
'line-3': 'string',
50+
},
51+
],
52+
},
53+
};
4154
```
4255
4356
@category Change case
4457
@category Template literal
4558
@category Object
4659
*/
47-
export type KebabCasedPropertiesDeep<Value> = DelimiterCasedPropertiesDeep<Value, '-'>;
60+
export type KebabCasedPropertiesDeep<
61+
Value,
62+
Options extends WordsOptions = {},
63+
> = DelimiterCasedPropertiesDeep<Value, '-', ApplyDefaultOptions<WordsOptions, DefaultDelimiterCaseOptions, Options>>;

source/kebab-cased-properties.d.ts

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,7 @@
1+
import type {DefaultDelimiterCaseOptions} from './delimiter-case';
12
import type {DelimiterCasedProperties} from './delimiter-cased-properties';
3+
import type {ApplyDefaultOptions} from './internal';
4+
import type {WordsOptions} from './words';
25

36
/**
47
Convert object properties to kebab case but not recursively.
@@ -21,10 +24,17 @@ const result: KebabCasedProperties<User> = {
2124
'user-id': 1,
2225
'user-name': 'Tom',
2326
};
27+
28+
const splitOnNumbers: KebabCasedProperties<{line1: string}, {splitOnNumbers: true}> = {
29+
'line-1': 'string',
30+
};
2431
```
2532
2633
@category Change case
2734
@category Template literal
2835
@category Object
2936
*/
30-
export type KebabCasedProperties<Value> = DelimiterCasedProperties<Value, '-'>;
37+
export type KebabCasedProperties<
38+
Value,
39+
Options extends WordsOptions = {},
40+
> = DelimiterCasedProperties<Value, '-', ApplyDefaultOptions<WordsOptions, DefaultDelimiterCaseOptions, Options>>;

source/snake-case.d.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,8 @@ import type {SnakeCase} from 'type-fest';
1414
// Simple
1515
1616
const someVariable: SnakeCase<'fooBar'> = 'foo_bar';
17-
const someVariableNoSplitOnNumbers: SnakeCase<'p2pNetwork', {splitOnNumbers: false}> = 'p2p_network';
17+
const noSplitOnNumbers: SnakeCase<'p2pNetwork'> = 'p2p_network';
18+
const splitOnNumbers: SnakeCase<'p2pNetwork', {splitOnNumbers: true}> = 'p_2_p_network';
1819
1920
// Advanced
2021

source/snake-cased-properties-deep.d.ts

+17-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,7 @@
1+
import type {DefaultDelimiterCaseOptions} from './delimiter-case';
12
import type {DelimiterCasedPropertiesDeep} from './delimiter-cased-properties-deep';
3+
import type {ApplyDefaultOptions} from './internal';
4+
import type {WordsOptions} from './words';
25

36
/**
47
Convert object properties to snake case recursively.
@@ -38,10 +41,23 @@ const result: SnakeCasedPropertiesDeep<UserWithFriends> = {
3841
},
3942
],
4043
};
44+
45+
const splitOnNumbers: SnakeCasedPropertiesDeep<{line1: { line2: [{ line3: string }] }}, {splitOnNumbers: true}> = {
46+
line_1: {
47+
line_2: [
48+
{
49+
line_3: 'string',
50+
},
51+
],
52+
},
53+
};
4154
```
4255
4356
@category Change case
4457
@category Template literal
4558
@category Object
4659
*/
47-
export type SnakeCasedPropertiesDeep<Value> = DelimiterCasedPropertiesDeep<Value, '_'>;
60+
export type SnakeCasedPropertiesDeep<
61+
Value,
62+
Options extends WordsOptions = {},
63+
> = DelimiterCasedPropertiesDeep<Value, '_', ApplyDefaultOptions<WordsOptions, DefaultDelimiterCaseOptions, Options>>;

source/snake-cased-properties.d.ts

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,7 @@
1+
import type {DefaultDelimiterCaseOptions} from './delimiter-case';
12
import type {DelimiterCasedProperties} from './delimiter-cased-properties';
3+
import type {ApplyDefaultOptions} from './internal';
4+
import type {WordsOptions} from './words';
25

36
/**
47
Convert object properties to snake case but not recursively.
@@ -21,10 +24,17 @@ const result: SnakeCasedProperties<User> = {
2124
user_id: 1,
2225
user_name: 'Tom',
2326
};
27+
28+
const splitOnNumbers: SnakeCasedProperties<{line1: string}, {splitOnNumbers: true}> = {
29+
'line_1': 'string',
30+
};
2431
```
2532
2633
@category Change case
2734
@category Template literal
2835
@category Object
2936
*/
30-
export type SnakeCasedProperties<Value> = DelimiterCasedProperties<Value, '_'>;
37+
export type SnakeCasedProperties<
38+
Value,
39+
Options extends WordsOptions = {},
40+
> = DelimiterCasedProperties<Value, '_', ApplyDefaultOptions<WordsOptions, DefaultDelimiterCaseOptions, Options>>;

test-d/delimiter-cased-properties-deep.ts

+3
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,9 @@ expectType<() => {a: string}>(fooBar);
1010
declare const bar: DelimiterCasedPropertiesDeep<Set<{fooBar: string}>, '-'>;
1111
expectType<Set<{'foo-bar': string}>>(bar);
1212

13+
declare const withOptions: DelimiterCasedPropertiesDeep<Set<{helloWorld: {p2p: Array<{addressLine1: string}>}}>, '.', {splitOnNumbers: true}>;
14+
expectType<Set<{'hello.world': {'p.2.p': Array<{'address.line.1': string}>}}>>(withOptions);
15+
1316
// Verify example
1417
type User = {
1518
userId: number;

test-d/delimiter-cased-properties.ts

+3
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,9 @@ expectType<Array<{helloWorld: string}>>(bar);
1010
declare const fooBar: DelimiterCasedProperties<() => {a: string}, '-'>;
1111
expectType<() => {a: string}>(fooBar);
1212

13+
declare const withOptions: DelimiterCasedProperties<{helloWorld1: {fooBar: string}}, '.', {splitOnNumbers: true}>;
14+
expectType<{'hello.world.1': {fooBar: string}}>(withOptions);
15+
1316
// Verify example
1417
type User = {
1518
userId: number;

test-d/kebab-cased-properties-deep.ts

+6-4
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,13 @@
11
import {expectType} from 'tsd';
22
import type {KebabCasedPropertiesDeep} from '../index';
33

4-
declare const foo: KebabCasedPropertiesDeep<{helloWorld: {fooBar: string}}>;
5-
expectType<{'hello-world': {'foo-bar': string}}>(foo);
4+
type FooBar = {helloWorld: {p2p: Array<{addressLine1: string}>}};
65

7-
declare const bar: KebabCasedPropertiesDeep<Set<{fooBar: string}>>;
8-
expectType<Set<{'foo-bar': string}>>(bar);
6+
declare const foo: KebabCasedPropertiesDeep<FooBar>;
7+
expectType<{'hello-world': {p2p: Array<{'address-line1': string}>}}>(foo);
8+
9+
declare const bar: KebabCasedPropertiesDeep<FooBar, {splitOnNumbers: true}>;
10+
expectType<{'hello-world': {'p-2-p': Array<{'address-line-1': string}>}}>(bar);
911

1012
// Verify example
1113
type User = {

test-d/kebab-cased-properties.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,13 @@
11
import {expectType} from 'tsd';
22
import type {KebabCasedProperties} from '../index';
33

4-
declare const foo: KebabCasedProperties<{helloWorld: {fooBar: string}}>;
5-
expectType<{'hello-world': {fooBar: string}}>(foo);
4+
type Foobar = {helloWorld1: {fooBar: string}};
5+
6+
declare const foo: KebabCasedProperties<Foobar>;
7+
expectType<{'hello-world1': {fooBar: string}}>(foo);
8+
9+
declare const bar: KebabCasedProperties<Foobar, {splitOnNumbers: true}>;
10+
expectType<{'hello-world-1': {fooBar: string}}>(bar);
611

712
// Verify example
813
type User = {

test-d/snake-cased-properties-deep.ts

+6-4
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,13 @@
11
import {expectType} from 'tsd';
22
import type {SnakeCasedPropertiesDeep} from '../index';
33

4-
declare const foo: SnakeCasedPropertiesDeep<{helloWorld: {fooBar: string}}>;
5-
expectType<{hello_world: {foo_bar: string}}>(foo);
4+
type FooBar = {helloWorld: {p2p: Array<{addressLine1: string}>}};
65

7-
declare const bar: SnakeCasedPropertiesDeep<Set<{fooBar: string}>>;
8-
expectType<Set<{foo_bar: string}>>(bar);
6+
declare const foo: SnakeCasedPropertiesDeep<FooBar>;
7+
expectType<{hello_world: {p2p: Array<{address_line1: string}>}}>(foo);
8+
9+
declare const bar: SnakeCasedPropertiesDeep<FooBar, {splitOnNumbers: true}>;
10+
expectType<{hello_world: {p_2_p: Array<{address_line_1: string}>}}>(bar);
911

1012
// Verify example
1113
type User = {

test-d/snake-cased-properties.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,13 @@
11
import {expectType} from 'tsd';
22
import type {SnakeCasedProperties} from '../index';
33

4-
declare const foo: SnakeCasedProperties<{helloWorld: {fooBar: string}}>;
5-
expectType<{hello_world: {fooBar: string}}>(foo);
4+
type Foobar = {helloWorld1: {fooBar: string}};
5+
6+
declare const foo: SnakeCasedProperties<Foobar>;
7+
expectType<{hello_world1: {fooBar: string}}>(foo);
8+
9+
declare const bar: SnakeCasedProperties<Foobar, {splitOnNumbers: true}>;
10+
expectType<{hello_world_1: {fooBar: string}}>(bar);
611

712
// Verify example
813
type User = {

0 commit comments

Comments
 (0)