Gets the name of a number, even if it's stupidly big. Supersedes TheoryOfNekomata/number-name.
 
 

118 linhas
2.8 KiB

  1. import { enUS } from './systems';
  2. import { StringifySystem } from './common';
  3. /**
  4. * Negative symbol.
  5. */
  6. const NEGATIVE_SYMBOL = '-' as const;
  7. /**
  8. * Allowed value type for {@link stringify}.
  9. */
  10. type AllowedValue = string | number | bigint;
  11. /**
  12. * Array of allowed types for {@link parse}.
  13. */
  14. const ALLOWED_PARSE_RESULT_TYPES = [
  15. 'string',
  16. 'number',
  17. 'bigint',
  18. ] as const;
  19. /**
  20. * Allowed type for {@link parse}.
  21. */
  22. type ParseResult = typeof ALLOWED_PARSE_RESULT_TYPES[number];
  23. /**
  24. * Options to use when converting a value to a string.
  25. */
  26. export interface StringifyOptions {
  27. /**
  28. * The system to use when converting a value to a string.
  29. *
  30. * Defaults to en-US (American short count).
  31. */
  32. system?: StringifySystem;
  33. /**
  34. * Options to use when making a group. This is used to override the default options for a group.
  35. */
  36. makeGroupOptions?: Record<string, unknown>;
  37. }
  38. /**
  39. * Converts a numeric value to its name.
  40. * @param value - The value to convert.
  41. * @param options - Options to use when converting a value to its name.
  42. * @returns The name of the value.
  43. */
  44. export const stringify = (
  45. value: AllowedValue,
  46. options = {} as StringifyOptions,
  47. ): string => {
  48. if (!(
  49. (ALLOWED_PARSE_RESULT_TYPES as unknown as string[])
  50. .includes(typeof (value as unknown))
  51. )) {
  52. throw new TypeError(`Value must be a string, number, or bigint. Received: ${typeof value}`);
  53. }
  54. const valueStr = value.toString().replace(/\s/g, '');
  55. const { system = enUS, makeGroupOptions } = options;
  56. if (valueStr.startsWith(NEGATIVE_SYMBOL)) {
  57. return system.makeNegative(stringify(valueStr.slice(NEGATIVE_SYMBOL.length), options));
  58. }
  59. const groups = system
  60. .group(valueStr)
  61. .map(([group, place]) => (
  62. system.makeGroup(group, place, makeGroupOptions)
  63. ));
  64. return system.finalize(groups);
  65. };
  66. /**
  67. * Options to use when parsing a name of a number to its numeric equivalent.
  68. */
  69. export interface ParseOptions {
  70. /**
  71. * The system to use when parsing a name of a number to its numeric equivalent.
  72. *
  73. * Defaults to en-US (American short count).
  74. */
  75. system?: StringifySystem;
  76. /**
  77. * The type to parse the value as.
  78. */
  79. type?: ParseResult;
  80. }
  81. /**
  82. * Parses a name of a number to its numeric equivalent.
  83. * @param value - The value to parse.
  84. * @param options - Options to use when parsing a name of a number to its numeric equivalent.
  85. * @returns The numeric equivalent of the value.
  86. */
  87. export const parse = (value: string, options = {} as ParseOptions) => {
  88. const { system = enUS, type = 'string' } = options;
  89. const tokens = system.tokenize(value);
  90. const groups = system.parseGroups(tokens);
  91. const stringValue = system.combineGroups(groups);
  92. switch (type) {
  93. case 'number':
  94. // Precision might be lost here. Use bigint when not using fractional parts.
  95. return Number(stringValue);
  96. case 'bigint':
  97. return BigInt(stringValue);
  98. default:
  99. break;
  100. }
  101. return stringValue;
  102. };