types: Allow choice's value type to be strictly inferred (#8529)

* types: stricter types for choices in options

* test: add choice tests

Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
This commit is contained in:
Jiralite
2022-08-20 21:29:16 +01:00
committed by GitHub
parent 0dba8adbd2
commit b3f7c32f7f
2 changed files with 31 additions and 12 deletions

View File

@@ -3873,19 +3873,21 @@ export interface ApplicationCommandAutocompleteStringOptionData
autocomplete: true; autocomplete: true;
} }
export interface ApplicationCommandChoicesData extends Omit<BaseApplicationCommandOptionsData, 'autocomplete'> { export interface ApplicationCommandChoicesData<Type extends string | number = string | number>
extends Omit<BaseApplicationCommandOptionsData, 'autocomplete'> {
type: CommandOptionChoiceResolvableType; type: CommandOptionChoiceResolvableType;
choices?: ApplicationCommandOptionChoiceData[]; choices?: ApplicationCommandOptionChoiceData<Type>[];
autocomplete?: false; autocomplete?: false;
} }
export interface ApplicationCommandChoicesOption extends Omit<BaseApplicationCommandOptionsData, 'autocomplete'> { export interface ApplicationCommandChoicesOption<Type extends string | number = string | number>
extends Omit<BaseApplicationCommandOptionsData, 'autocomplete'> {
type: CommandOptionChoiceResolvableType; type: CommandOptionChoiceResolvableType;
choices?: ApplicationCommandOptionChoiceData[]; choices?: ApplicationCommandOptionChoiceData<Type>[];
autocomplete?: false; autocomplete?: false;
} }
export interface ApplicationCommandNumericOptionData extends ApplicationCommandChoicesData { export interface ApplicationCommandNumericOptionData extends ApplicationCommandChoicesData<number> {
type: CommandOptionNumericResolvableType; type: CommandOptionNumericResolvableType;
minValue?: number; minValue?: number;
min_value?: number; min_value?: number;
@@ -3893,7 +3895,7 @@ export interface ApplicationCommandNumericOptionData extends ApplicationCommandC
max_value?: number; max_value?: number;
} }
export interface ApplicationCommandStringOptionData extends ApplicationCommandChoicesData { export interface ApplicationCommandStringOptionData extends ApplicationCommandChoicesData<string> {
type: ApplicationCommandOptionType.String; type: ApplicationCommandOptionType.String;
minLength?: number; minLength?: number;
min_length?: number; min_length?: number;
@@ -3905,13 +3907,13 @@ export interface ApplicationCommandBooleanOptionData extends BaseApplicationComm
type: ApplicationCommandOptionType.Boolean; type: ApplicationCommandOptionType.Boolean;
} }
export interface ApplicationCommandNumericOption extends ApplicationCommandChoicesOption { export interface ApplicationCommandNumericOption extends ApplicationCommandChoicesOption<number> {
type: CommandOptionNumericResolvableType; type: CommandOptionNumericResolvableType;
minValue?: number; minValue?: number;
maxValue?: number; maxValue?: number;
} }
export interface ApplicationCommandStringOption extends ApplicationCommandChoicesOption { export interface ApplicationCommandStringOption extends ApplicationCommandChoicesOption<string> {
type: ApplicationCommandOptionType.String; type: ApplicationCommandOptionType.String;
minLength?: number; minLength?: number;
maxLength?: number; maxLength?: number;
@@ -3953,7 +3955,6 @@ export type ApplicationCommandOptionData =
| ApplicationCommandSubGroupData | ApplicationCommandSubGroupData
| ApplicationCommandNonOptionsData | ApplicationCommandNonOptionsData
| ApplicationCommandChannelOptionData | ApplicationCommandChannelOptionData
| ApplicationCommandChoicesData
| ApplicationCommandAutocompleteNumericOptionData | ApplicationCommandAutocompleteNumericOptionData
| ApplicationCommandAutocompleteStringOptionData | ApplicationCommandAutocompleteStringOptionData
| ApplicationCommandNumericOptionData | ApplicationCommandNumericOptionData
@@ -3970,7 +3971,6 @@ export type ApplicationCommandOption =
| ApplicationCommandAutocompleteStringOption | ApplicationCommandAutocompleteStringOption
| ApplicationCommandNonOptions | ApplicationCommandNonOptions
| ApplicationCommandChannelOption | ApplicationCommandChannelOption
| ApplicationCommandChoicesOption
| ApplicationCommandNumericOption | ApplicationCommandNumericOption
| ApplicationCommandStringOption | ApplicationCommandStringOption
| ApplicationCommandRoleOption | ApplicationCommandRoleOption
@@ -3980,10 +3980,10 @@ export type ApplicationCommandOption =
| ApplicationCommandAttachmentOption | ApplicationCommandAttachmentOption
| ApplicationCommandSubCommand; | ApplicationCommandSubCommand;
export interface ApplicationCommandOptionChoiceData { export interface ApplicationCommandOptionChoiceData<Value extends string | number = string | number> {
name: string; name: string;
nameLocalizations?: LocalizationMap; nameLocalizations?: LocalizationMap;
value: string | number; value: Value;
} }
export interface ApplicationCommandPermissions { export interface ApplicationCommandPermissions {

View File

@@ -1102,6 +1102,7 @@ expectAssignable<'death'>(ShardEvents.Death);
expectAssignable<1>(Status.Connecting); expectAssignable<1>(Status.Connecting);
declare const applicationCommandData: ApplicationCommandData; declare const applicationCommandData: ApplicationCommandData;
declare const applicationCommandOptionData: ApplicationCommandOptionData;
declare const applicationCommandResolvable: ApplicationCommandResolvable; declare const applicationCommandResolvable: ApplicationCommandResolvable;
declare const applicationCommandManager: ApplicationCommandManager; declare const applicationCommandManager: ApplicationCommandManager;
{ {
@@ -1121,6 +1122,24 @@ declare const applicationCommandManager: ApplicationCommandManager;
expectType<Promise<Collection<Snowflake, ApplicationCommand>>>( expectType<Promise<Collection<Snowflake, ApplicationCommand>>>(
applicationCommandManager.set([applicationCommandData], '0'), applicationCommandManager.set([applicationCommandData], '0'),
); );
// Test inference of choice values.
if ('choices' in applicationCommandOptionData) {
if (applicationCommandOptionData.type === ApplicationCommandOptionType.String) {
expectType<string>(applicationCommandOptionData.choices[0]!.value);
expectNotType<number>(applicationCommandOptionData.choices[0]!.value);
}
if (applicationCommandOptionData.type === ApplicationCommandOptionType.Integer) {
expectType<number>(applicationCommandOptionData.choices[0]!.value);
expectNotType<string>(applicationCommandOptionData.choices[0]!.value);
}
if (applicationCommandOptionData.type === ApplicationCommandOptionType.Number) {
expectType<number>(applicationCommandOptionData.choices[0]!.value);
expectNotType<string>(applicationCommandOptionData.choices[0]!.value);
}
}
} }
declare const applicationNonChoiceOptionData: ApplicationCommandOptionData & { declare const applicationNonChoiceOptionData: ApplicationCommandOptionData & {