n2words 2.0.0 → 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (335) hide show
  1. package/CHANGELOG.md +64 -0
  2. package/README.md +86 -188
  3. package/dist/languages/am-Latn.js +3 -0
  4. package/dist/languages/am-Latn.js.map +1 -0
  5. package/dist/languages/am.js +3 -0
  6. package/dist/languages/am.js.map +1 -0
  7. package/dist/languages/ar.js +3 -0
  8. package/dist/languages/ar.js.map +1 -0
  9. package/dist/languages/az.js +3 -0
  10. package/dist/languages/az.js.map +1 -0
  11. package/dist/languages/bn.js +3 -0
  12. package/dist/languages/bn.js.map +1 -0
  13. package/dist/languages/cs.js +3 -0
  14. package/dist/languages/cs.js.map +1 -0
  15. package/dist/languages/da.js +3 -0
  16. package/dist/languages/da.js.map +1 -0
  17. package/dist/languages/de.js +3 -0
  18. package/dist/languages/de.js.map +1 -0
  19. package/dist/languages/el.js +3 -0
  20. package/dist/languages/el.js.map +1 -0
  21. package/dist/languages/en.js +3 -0
  22. package/dist/languages/en.js.map +1 -0
  23. package/dist/languages/es.js +3 -0
  24. package/dist/languages/es.js.map +1 -0
  25. package/dist/languages/fa.js +3 -0
  26. package/dist/languages/fa.js.map +1 -0
  27. package/dist/languages/fi.js +3 -0
  28. package/dist/languages/fi.js.map +1 -0
  29. package/dist/languages/fil.js +3 -0
  30. package/dist/languages/fil.js.map +1 -0
  31. package/dist/languages/fr-BE.js +3 -0
  32. package/dist/languages/fr-BE.js.map +1 -0
  33. package/dist/languages/fr.js +3 -0
  34. package/dist/languages/fr.js.map +1 -0
  35. package/dist/languages/gu.js +3 -0
  36. package/dist/languages/gu.js.map +1 -0
  37. package/dist/languages/ha.js +3 -0
  38. package/dist/languages/ha.js.map +1 -0
  39. package/dist/languages/hbo.js +3 -0
  40. package/dist/languages/hbo.js.map +1 -0
  41. package/dist/languages/he.js +3 -0
  42. package/dist/languages/he.js.map +1 -0
  43. package/dist/languages/hi.js +3 -0
  44. package/dist/languages/hi.js.map +1 -0
  45. package/dist/languages/hr.js +3 -0
  46. package/dist/languages/hr.js.map +1 -0
  47. package/dist/languages/hu.js +3 -0
  48. package/dist/languages/hu.js.map +1 -0
  49. package/dist/languages/id.js +3 -0
  50. package/dist/languages/id.js.map +1 -0
  51. package/dist/languages/it.js +3 -0
  52. package/dist/languages/it.js.map +1 -0
  53. package/dist/languages/ja.js +3 -0
  54. package/dist/languages/ja.js.map +1 -0
  55. package/dist/languages/ka.js +3 -0
  56. package/dist/languages/ka.js.map +1 -0
  57. package/dist/languages/kn.js +3 -0
  58. package/dist/languages/kn.js.map +1 -0
  59. package/dist/languages/ko.js +3 -0
  60. package/dist/languages/ko.js.map +1 -0
  61. package/dist/languages/lt.js +3 -0
  62. package/dist/languages/lt.js.map +1 -0
  63. package/dist/languages/lv.js +3 -0
  64. package/dist/languages/lv.js.map +1 -0
  65. package/dist/languages/mr.js +3 -0
  66. package/dist/languages/mr.js.map +1 -0
  67. package/dist/languages/ms.js +3 -0
  68. package/dist/languages/ms.js.map +1 -0
  69. package/dist/languages/nb.js +3 -0
  70. package/dist/languages/nb.js.map +1 -0
  71. package/dist/languages/nl.js +3 -0
  72. package/dist/languages/nl.js.map +1 -0
  73. package/dist/languages/pa.js +3 -0
  74. package/dist/languages/pa.js.map +1 -0
  75. package/dist/languages/pl.js +3 -0
  76. package/dist/languages/pl.js.map +1 -0
  77. package/dist/languages/pt.js +3 -0
  78. package/dist/languages/pt.js.map +1 -0
  79. package/dist/languages/ro.js +3 -0
  80. package/dist/languages/ro.js.map +1 -0
  81. package/dist/languages/ru.js +3 -0
  82. package/dist/languages/ru.js.map +1 -0
  83. package/dist/languages/sr-Cyrl.js +3 -0
  84. package/dist/languages/sr-Cyrl.js.map +1 -0
  85. package/dist/languages/sr-Latn.js +3 -0
  86. package/dist/languages/sr-Latn.js.map +1 -0
  87. package/dist/languages/sv.js +3 -0
  88. package/dist/languages/sv.js.map +1 -0
  89. package/dist/languages/sw.js +3 -0
  90. package/dist/languages/sw.js.map +1 -0
  91. package/dist/languages/ta.js +3 -0
  92. package/dist/languages/ta.js.map +1 -0
  93. package/dist/languages/te.js +3 -0
  94. package/dist/languages/te.js.map +1 -0
  95. package/dist/languages/th.js +3 -0
  96. package/dist/languages/th.js.map +1 -0
  97. package/dist/languages/tr.js +3 -0
  98. package/dist/languages/tr.js.map +1 -0
  99. package/dist/languages/uk.js +3 -0
  100. package/dist/languages/uk.js.map +1 -0
  101. package/dist/languages/ur.js +3 -0
  102. package/dist/languages/ur.js.map +1 -0
  103. package/dist/languages/vi.js +3 -0
  104. package/dist/languages/vi.js.map +1 -0
  105. package/dist/languages/yo.js +3 -0
  106. package/dist/languages/yo.js.map +1 -0
  107. package/dist/languages/zh-Hans.js +3 -0
  108. package/dist/languages/zh-Hans.js.map +1 -0
  109. package/dist/languages/zh-Hant.js +3 -0
  110. package/dist/languages/zh-Hant.js.map +1 -0
  111. package/dist/n2words.js +2 -2
  112. package/dist/n2words.js.map +1 -1
  113. package/lib/languages/am-Latn.d.ts +7 -0
  114. package/lib/languages/am-Latn.js +159 -0
  115. package/lib/languages/am.d.ts +7 -0
  116. package/lib/languages/am.js +159 -0
  117. package/lib/languages/ar.d.ts +14 -27
  118. package/lib/languages/ar.js +175 -129
  119. package/lib/languages/az.d.ts +4 -9
  120. package/lib/languages/az.js +166 -37
  121. package/lib/languages/bn.d.ts +4 -8
  122. package/lib/languages/bn.js +159 -124
  123. package/lib/languages/cs.d.ts +15 -85
  124. package/lib/languages/cs.js +293 -114
  125. package/lib/languages/da.d.ts +11 -12
  126. package/lib/languages/da.js +269 -101
  127. package/lib/languages/de.d.ts +14 -11
  128. package/lib/languages/de.js +305 -86
  129. package/lib/languages/el.d.ts +11 -11
  130. package/lib/languages/el.js +224 -78
  131. package/lib/languages/en.d.ts +14 -13
  132. package/lib/languages/en.js +228 -72
  133. package/lib/languages/es.d.ts +18 -12
  134. package/lib/languages/es.js +297 -103
  135. package/lib/languages/fa.d.ts +4 -44
  136. package/lib/languages/fa.js +112 -122
  137. package/lib/languages/fi.d.ts +14 -0
  138. package/lib/languages/fi.js +238 -0
  139. package/lib/languages/fil.d.ts +4 -13
  140. package/lib/languages/fil.js +196 -106
  141. package/lib/languages/fr-BE.d.ts +8 -8
  142. package/lib/languages/fr-BE.js +285 -19
  143. package/lib/languages/fr.d.ts +18 -12
  144. package/lib/languages/fr.js +339 -89
  145. package/lib/languages/gu.d.ts +4 -8
  146. package/lib/languages/gu.js +143 -125
  147. package/lib/languages/ha.d.ts +7 -0
  148. package/lib/languages/ha.js +225 -0
  149. package/lib/languages/hbo.d.ts +10 -110
  150. package/lib/languages/hbo.js +245 -214
  151. package/lib/languages/he.d.ts +10 -77
  152. package/lib/languages/he.js +231 -172
  153. package/lib/languages/hi.d.ts +4 -8
  154. package/lib/languages/hi.js +163 -124
  155. package/lib/languages/hr.d.ts +8 -77
  156. package/lib/languages/hr.js +200 -89
  157. package/lib/languages/hu.d.ts +4 -19
  158. package/lib/languages/hu.js +198 -119
  159. package/lib/languages/id.d.ts +4 -34
  160. package/lib/languages/id.js +166 -129
  161. package/lib/languages/it.d.ts +16 -34
  162. package/lib/languages/it.js +307 -97
  163. package/lib/languages/ja.d.ts +14 -14
  164. package/lib/languages/ja.js +221 -111
  165. package/lib/languages/ka.d.ts +17 -0
  166. package/lib/languages/ka.js +291 -0
  167. package/lib/languages/kn.d.ts +4 -8
  168. package/lib/languages/kn.js +143 -35
  169. package/lib/languages/ko.d.ts +11 -11
  170. package/lib/languages/ko.js +250 -49
  171. package/lib/languages/lt.d.ts +15 -67
  172. package/lib/languages/lt.js +287 -122
  173. package/lib/languages/lv.d.ts +15 -67
  174. package/lib/languages/lv.js +288 -106
  175. package/lib/languages/mr.d.ts +4 -8
  176. package/lib/languages/mr.js +143 -125
  177. package/lib/languages/ms.d.ts +4 -28
  178. package/lib/languages/ms.js +166 -116
  179. package/lib/languages/nb.d.ts +11 -9
  180. package/lib/languages/nb.js +272 -87
  181. package/lib/languages/nl.d.ts +23 -13
  182. package/lib/languages/nl.js +299 -133
  183. package/lib/languages/pa.d.ts +4 -8
  184. package/lib/languages/pa.js +151 -124
  185. package/lib/languages/pl.d.ts +19 -77
  186. package/lib/languages/pl.js +294 -87
  187. package/lib/languages/pt.d.ts +14 -26
  188. package/lib/languages/pt.js +272 -92
  189. package/lib/languages/ro.d.ts +15 -155
  190. package/lib/languages/ro.js +219 -235
  191. package/lib/languages/ru.d.ts +8 -82
  192. package/lib/languages/ru.js +239 -90
  193. package/lib/languages/sr-Cyrl.d.ts +8 -77
  194. package/lib/languages/sr-Cyrl.js +197 -89
  195. package/lib/languages/sr-Latn.d.ts +8 -77
  196. package/lib/languages/sr-Latn.js +197 -89
  197. package/lib/languages/sv.d.ts +11 -11
  198. package/lib/languages/sv.js +278 -74
  199. package/lib/languages/sw.d.ts +4 -36
  200. package/lib/languages/sw.js +133 -106
  201. package/lib/languages/ta.d.ts +4 -17
  202. package/lib/languages/ta.js +143 -202
  203. package/lib/languages/te.d.ts +4 -19
  204. package/lib/languages/te.js +133 -196
  205. package/lib/languages/th.d.ts +4 -14
  206. package/lib/languages/th.js +135 -91
  207. package/lib/languages/tr.d.ts +15 -9
  208. package/lib/languages/tr.js +245 -49
  209. package/lib/languages/uk.d.ts +8 -82
  210. package/lib/languages/uk.js +206 -78
  211. package/lib/languages/ur.d.ts +4 -8
  212. package/lib/languages/ur.js +151 -124
  213. package/lib/languages/vi.d.ts +14 -69
  214. package/lib/languages/vi.js +278 -129
  215. package/lib/languages/yo.d.ts +7 -0
  216. package/lib/languages/yo.js +303 -0
  217. package/lib/languages/zh-Hans.d.ts +8 -18
  218. package/lib/languages/zh-Hans.js +163 -92
  219. package/lib/languages/zh-Hant.d.ts +8 -18
  220. package/lib/languages/zh-Hant.js +181 -90
  221. package/lib/n2words.d.ts +55 -209
  222. package/lib/n2words.js +115 -530
  223. package/lib/utils/is-plain-object.d.ts +13 -0
  224. package/lib/utils/is-plain-object.js +17 -0
  225. package/lib/utils/parse-numeric.d.ts +17 -0
  226. package/lib/utils/parse-numeric.js +108 -0
  227. package/lib/utils/validate-options.d.ts +8 -0
  228. package/lib/utils/validate-options.js +16 -0
  229. package/package.json +26 -14
  230. package/dist/ArabicConverter.js +0 -3
  231. package/dist/ArabicConverter.js.map +0 -1
  232. package/dist/AzerbaijaniConverter.js +0 -3
  233. package/dist/AzerbaijaniConverter.js.map +0 -1
  234. package/dist/BanglaConverter.js +0 -3
  235. package/dist/BanglaConverter.js.map +0 -1
  236. package/dist/BiblicalHebrewConverter.js +0 -3
  237. package/dist/BiblicalHebrewConverter.js.map +0 -1
  238. package/dist/CroatianConverter.js +0 -3
  239. package/dist/CroatianConverter.js.map +0 -1
  240. package/dist/CzechConverter.js +0 -3
  241. package/dist/CzechConverter.js.map +0 -1
  242. package/dist/DanishConverter.js +0 -3
  243. package/dist/DanishConverter.js.map +0 -1
  244. package/dist/DutchConverter.js +0 -3
  245. package/dist/DutchConverter.js.map +0 -1
  246. package/dist/EnglishConverter.js +0 -3
  247. package/dist/EnglishConverter.js.map +0 -1
  248. package/dist/FilipinoConverter.js +0 -3
  249. package/dist/FilipinoConverter.js.map +0 -1
  250. package/dist/FrenchBelgiumConverter.js +0 -3
  251. package/dist/FrenchBelgiumConverter.js.map +0 -1
  252. package/dist/FrenchConverter.js +0 -3
  253. package/dist/FrenchConverter.js.map +0 -1
  254. package/dist/GermanConverter.js +0 -3
  255. package/dist/GermanConverter.js.map +0 -1
  256. package/dist/GreekConverter.js +0 -3
  257. package/dist/GreekConverter.js.map +0 -1
  258. package/dist/GujaratiConverter.js +0 -3
  259. package/dist/GujaratiConverter.js.map +0 -1
  260. package/dist/HebrewConverter.js +0 -3
  261. package/dist/HebrewConverter.js.map +0 -1
  262. package/dist/HindiConverter.js +0 -3
  263. package/dist/HindiConverter.js.map +0 -1
  264. package/dist/HungarianConverter.js +0 -3
  265. package/dist/HungarianConverter.js.map +0 -1
  266. package/dist/IndonesianConverter.js +0 -3
  267. package/dist/IndonesianConverter.js.map +0 -1
  268. package/dist/ItalianConverter.js +0 -3
  269. package/dist/ItalianConverter.js.map +0 -1
  270. package/dist/JapaneseConverter.js +0 -3
  271. package/dist/JapaneseConverter.js.map +0 -1
  272. package/dist/KannadaConverter.js +0 -3
  273. package/dist/KannadaConverter.js.map +0 -1
  274. package/dist/KoreanConverter.js +0 -3
  275. package/dist/KoreanConverter.js.map +0 -1
  276. package/dist/LatvianConverter.js +0 -3
  277. package/dist/LatvianConverter.js.map +0 -1
  278. package/dist/LithuanianConverter.js +0 -3
  279. package/dist/LithuanianConverter.js.map +0 -1
  280. package/dist/MalayConverter.js +0 -3
  281. package/dist/MalayConverter.js.map +0 -1
  282. package/dist/MarathiConverter.js +0 -3
  283. package/dist/MarathiConverter.js.map +0 -1
  284. package/dist/NorwegianBokmalConverter.js +0 -3
  285. package/dist/NorwegianBokmalConverter.js.map +0 -1
  286. package/dist/PersianConverter.js +0 -3
  287. package/dist/PersianConverter.js.map +0 -1
  288. package/dist/PolishConverter.js +0 -3
  289. package/dist/PolishConverter.js.map +0 -1
  290. package/dist/PortugueseConverter.js +0 -3
  291. package/dist/PortugueseConverter.js.map +0 -1
  292. package/dist/PunjabiConverter.js +0 -3
  293. package/dist/PunjabiConverter.js.map +0 -1
  294. package/dist/RomanianConverter.js +0 -3
  295. package/dist/RomanianConverter.js.map +0 -1
  296. package/dist/RussianConverter.js +0 -3
  297. package/dist/RussianConverter.js.map +0 -1
  298. package/dist/SerbianCyrillicConverter.js +0 -3
  299. package/dist/SerbianCyrillicConverter.js.map +0 -1
  300. package/dist/SerbianLatinConverter.js +0 -3
  301. package/dist/SerbianLatinConverter.js.map +0 -1
  302. package/dist/SimplifiedChineseConverter.js +0 -3
  303. package/dist/SimplifiedChineseConverter.js.map +0 -1
  304. package/dist/SpanishConverter.js +0 -3
  305. package/dist/SpanishConverter.js.map +0 -1
  306. package/dist/SwahiliConverter.js +0 -3
  307. package/dist/SwahiliConverter.js.map +0 -1
  308. package/dist/SwedishConverter.js +0 -3
  309. package/dist/SwedishConverter.js.map +0 -1
  310. package/dist/TamilConverter.js +0 -3
  311. package/dist/TamilConverter.js.map +0 -1
  312. package/dist/TeluguConverter.js +0 -3
  313. package/dist/TeluguConverter.js.map +0 -1
  314. package/dist/ThaiConverter.js +0 -3
  315. package/dist/ThaiConverter.js.map +0 -1
  316. package/dist/TraditionalChineseConverter.js +0 -3
  317. package/dist/TraditionalChineseConverter.js.map +0 -1
  318. package/dist/TurkishConverter.js +0 -3
  319. package/dist/TurkishConverter.js.map +0 -1
  320. package/dist/UkrainianConverter.js +0 -3
  321. package/dist/UkrainianConverter.js.map +0 -1
  322. package/dist/UrduConverter.js +0 -3
  323. package/dist/UrduConverter.js.map +0 -1
  324. package/dist/VietnameseConverter.js +0 -3
  325. package/dist/VietnameseConverter.js.map +0 -1
  326. package/lib/classes/abstract-language.d.ts +0 -178
  327. package/lib/classes/abstract-language.js +0 -268
  328. package/lib/classes/greedy-scale-language.d.ts +0 -109
  329. package/lib/classes/greedy-scale-language.js +0 -201
  330. package/lib/classes/slavic-language.d.ts +0 -148
  331. package/lib/classes/slavic-language.js +0 -281
  332. package/lib/classes/south-asian-language.d.ts +0 -70
  333. package/lib/classes/south-asian-language.js +0 -154
  334. package/lib/classes/turkic-language.d.ts +0 -26
  335. package/lib/classes/turkic-language.js +0 -59
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Converts a numeric value to Amharic (Latin script) words.
3
+ *
4
+ * @param {number | string | bigint} value - The numeric value to convert
5
+ * @returns {string} The number in Amharic Latin words
6
+ */
7
+ export function toWords(value: number | string | bigint): string;
@@ -0,0 +1,159 @@
1
+ /**
2
+ * Amharic Latin language converter - Functional Implementation
3
+ *
4
+ * Self-contained module with its own input validation, ready for subpath exports.
5
+ * Latin/ASCII romanization of Amharic numerals.
6
+ *
7
+ * Key features:
8
+ * - Romanized numerals (and, hulet, sost)
9
+ * - Teens formed with "asra" prefix
10
+ * - Keeps "one" before hundred: "and meto" (100)
11
+ * - Short scale naming
12
+ * - Per-digit decimal reading
13
+ */
14
+
15
+ import { parseNumericValue } from '../utils/parse-numeric.js'
16
+
17
+ // ============================================================================
18
+ // Vocabulary
19
+ // ============================================================================
20
+
21
+ const ONES = ['', 'and', 'hulet', 'sost', 'arat', 'amist', 'siddist', 'sebat', 'siment', 'zeteny']
22
+ const TEENS = ['asir', 'asra and', 'asra hulet', 'asra sost', 'asra arat', 'asra amist', 'asra siddist', 'asra sebat', 'asra siment', 'asra zeteny']
23
+ const TENS = ['', '', 'haya', 'selasa', 'arba', 'hamsa', 'silsa', 'seba', 'semanya', 'zetena']
24
+
25
+ const HUNDRED = 'meto'
26
+ const THOUSAND = 'shi'
27
+
28
+ const ZERO = 'zero'
29
+ const NEGATIVE = 'asitegna'
30
+ const DECIMAL_SEP = 'netib'
31
+
32
+ // Short scale
33
+ const SCALE_WORDS = ['', THOUSAND, 'miliyon', 'billiyon']
34
+
35
+ // ============================================================================
36
+ // Precomputed Lookup Table
37
+ // ============================================================================
38
+
39
+ function buildSegment (n) {
40
+ if (n === 0) return ''
41
+
42
+ const ones = n % 10
43
+ const tensDigit = Math.floor(n / 10) % 10
44
+ const hundredsDigit = Math.floor(n / 100)
45
+
46
+ const parts = []
47
+
48
+ // Amharic keeps "one" before hundred: "and meto" (100)
49
+ if (hundredsDigit > 0) {
50
+ parts.push(ONES[hundredsDigit] + ' ' + HUNDRED)
51
+ }
52
+
53
+ if (tensDigit === 1) {
54
+ parts.push(TEENS[ones])
55
+ } else {
56
+ if (tensDigit > 1) {
57
+ parts.push(TENS[tensDigit])
58
+ }
59
+ if (ones > 0) {
60
+ parts.push(ONES[ones])
61
+ }
62
+ }
63
+
64
+ return parts.join(' ')
65
+ }
66
+
67
+ // ============================================================================
68
+ // Conversion Functions
69
+ // ============================================================================
70
+
71
+ function integerToWords (n) {
72
+ if (n === 0n) return ZERO
73
+
74
+ if (n < 1000n) {
75
+ return buildSegment(Number(n))
76
+ }
77
+
78
+ return buildLargeNumberWords(n)
79
+ }
80
+
81
+ function buildLargeNumberWords (n) {
82
+ const numStr = n.toString()
83
+ const len = numStr.length
84
+
85
+ const segments = []
86
+ const segmentSize = 3
87
+
88
+ const remainderLen = len % segmentSize
89
+ let pos = 0
90
+ if (remainderLen > 0) {
91
+ segments.push(Number(numStr.slice(0, remainderLen)))
92
+ pos = remainderLen
93
+ }
94
+ while (pos < len) {
95
+ segments.push(Number(numStr.slice(pos, pos + segmentSize)))
96
+ pos += segmentSize
97
+ }
98
+
99
+ const parts = []
100
+ let scaleIndex = segments.length - 1
101
+
102
+ for (let i = 0; i < segments.length; i++) {
103
+ const segment = segments[i]
104
+
105
+ if (segment !== 0) {
106
+ const scaleWord = SCALE_WORDS[scaleIndex] || ''
107
+
108
+ if (scaleIndex === 0) {
109
+ parts.push(buildSegment(segment))
110
+ } else {
111
+ parts.push(buildSegment(segment) + ' ' + scaleWord)
112
+ }
113
+ }
114
+
115
+ scaleIndex--
116
+ }
117
+
118
+ return parts.join(' ')
119
+ }
120
+
121
+ function decimalPartToWords (decimalPart) {
122
+ // Per-digit decimal reading
123
+ const digits = []
124
+ for (const char of decimalPart) {
125
+ const d = parseInt(char, 10)
126
+ digits.push(d === 0 ? ZERO : ONES[d])
127
+ }
128
+ return digits.join(' ')
129
+ }
130
+
131
+ /**
132
+ * Converts a numeric value to Amharic (Latin script) words.
133
+ *
134
+ * @param {number | string | bigint} value - The numeric value to convert
135
+ * @returns {string} The number in Amharic Latin words
136
+ */
137
+ function toWords (value) {
138
+ const { isNegative, integerPart, decimalPart } = parseNumericValue(value)
139
+
140
+ let result = ''
141
+
142
+ if (isNegative) {
143
+ result = NEGATIVE + ' '
144
+ }
145
+
146
+ result += integerToWords(integerPart)
147
+
148
+ if (decimalPart) {
149
+ result += ' ' + DECIMAL_SEP + ' ' + decimalPartToWords(decimalPart)
150
+ }
151
+
152
+ return result
153
+ }
154
+
155
+ // ============================================================================
156
+ // Exports
157
+ // ============================================================================
158
+
159
+ export { toWords }
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Converts a numeric value to Amharic words.
3
+ *
4
+ * @param {number | string | bigint} value - The numeric value to convert
5
+ * @returns {string} The number in Amharic words
6
+ */
7
+ export function toWords(value: number | string | bigint): string;
@@ -0,0 +1,159 @@
1
+ /**
2
+ * Amharic language converter - Functional Implementation
3
+ *
4
+ * Self-contained module with its own input validation, ready for subpath exports.
5
+ * Native Ge'ez script (ግዕዝ) output.
6
+ *
7
+ * Key features:
8
+ * - Ge'ez/Ethiopic script numerals
9
+ * - Teens formed with "አስራ" prefix
10
+ * - Keeps "one" before hundred: "አንድ መቶ" (100)
11
+ * - Short scale naming
12
+ * - Per-digit decimal reading
13
+ */
14
+
15
+ import { parseNumericValue } from '../utils/parse-numeric.js'
16
+
17
+ // ============================================================================
18
+ // Vocabulary
19
+ // ============================================================================
20
+
21
+ const ONES = ['', 'አንድ', 'ሁለት', 'ሶስት', 'አራት', 'አምስት', 'ስድስት', 'ሰባት', 'ስምንት', 'ዘጠኝ']
22
+ const TEENS = ['አስር', 'አስራ አንድ', 'አስራ ሁለት', 'አስራ ሶስት', 'አስራ አራት', 'አስራ አምስት', 'አስራ ስድስት', 'አስራ ሰባት', 'አስራ ስምንት', 'አስራ ዘጠኝ']
23
+ const TENS = ['', '', 'ሃያ', 'ሰላሳ', 'አርባ', 'ሃምሳ', 'ስልሳ', 'ሰባ', 'ሰማንያ', 'ዘጠና']
24
+
25
+ const HUNDRED = 'መቶ'
26
+ const THOUSAND = 'ሺ'
27
+
28
+ const ZERO = 'ዜሮ'
29
+ const NEGATIVE = 'አሉታዊ'
30
+ const DECIMAL_SEP = 'ነጥብ'
31
+
32
+ // Short scale
33
+ const SCALE_WORDS = ['', THOUSAND, 'ሚሊዮን', 'ቢሊዮን']
34
+
35
+ // ============================================================================
36
+ // Precomputed Lookup Table
37
+ // ============================================================================
38
+
39
+ function buildSegment (n) {
40
+ if (n === 0) return ''
41
+
42
+ const ones = n % 10
43
+ const tensDigit = Math.floor(n / 10) % 10
44
+ const hundredsDigit = Math.floor(n / 100)
45
+
46
+ const parts = []
47
+
48
+ // Amharic keeps "one" before hundred: "አንድ መቶ" (100)
49
+ if (hundredsDigit > 0) {
50
+ parts.push(ONES[hundredsDigit] + ' ' + HUNDRED)
51
+ }
52
+
53
+ if (tensDigit === 1) {
54
+ parts.push(TEENS[ones])
55
+ } else {
56
+ if (tensDigit > 1) {
57
+ parts.push(TENS[tensDigit])
58
+ }
59
+ if (ones > 0) {
60
+ parts.push(ONES[ones])
61
+ }
62
+ }
63
+
64
+ return parts.join(' ')
65
+ }
66
+
67
+ // ============================================================================
68
+ // Conversion Functions
69
+ // ============================================================================
70
+
71
+ function integerToWords (n) {
72
+ if (n === 0n) return ZERO
73
+
74
+ if (n < 1000n) {
75
+ return buildSegment(Number(n))
76
+ }
77
+
78
+ return buildLargeNumberWords(n)
79
+ }
80
+
81
+ function buildLargeNumberWords (n) {
82
+ const numStr = n.toString()
83
+ const len = numStr.length
84
+
85
+ const segments = []
86
+ const segmentSize = 3
87
+
88
+ const remainderLen = len % segmentSize
89
+ let pos = 0
90
+ if (remainderLen > 0) {
91
+ segments.push(Number(numStr.slice(0, remainderLen)))
92
+ pos = remainderLen
93
+ }
94
+ while (pos < len) {
95
+ segments.push(Number(numStr.slice(pos, pos + segmentSize)))
96
+ pos += segmentSize
97
+ }
98
+
99
+ const parts = []
100
+ let scaleIndex = segments.length - 1
101
+
102
+ for (let i = 0; i < segments.length; i++) {
103
+ const segment = segments[i]
104
+
105
+ if (segment !== 0) {
106
+ const scaleWord = SCALE_WORDS[scaleIndex] || ''
107
+
108
+ if (scaleIndex === 0) {
109
+ parts.push(buildSegment(segment))
110
+ } else {
111
+ parts.push(buildSegment(segment) + ' ' + scaleWord)
112
+ }
113
+ }
114
+
115
+ scaleIndex--
116
+ }
117
+
118
+ return parts.join(' ')
119
+ }
120
+
121
+ function decimalPartToWords (decimalPart) {
122
+ // Per-digit decimal reading
123
+ const digits = []
124
+ for (const char of decimalPart) {
125
+ const d = parseInt(char, 10)
126
+ digits.push(d === 0 ? ZERO : ONES[d])
127
+ }
128
+ return digits.join(' ')
129
+ }
130
+
131
+ /**
132
+ * Converts a numeric value to Amharic words.
133
+ *
134
+ * @param {number | string | bigint} value - The numeric value to convert
135
+ * @returns {string} The number in Amharic words
136
+ */
137
+ function toWords (value) {
138
+ const { isNegative, integerPart, decimalPart } = parseNumericValue(value)
139
+
140
+ let result = ''
141
+
142
+ if (isNegative) {
143
+ result = NEGATIVE + ' '
144
+ }
145
+
146
+ result += integerToWords(integerPart)
147
+
148
+ if (decimalPart) {
149
+ result += ' ' + DECIMAL_SEP + ' ' + decimalPartToWords(decimalPart)
150
+ }
151
+
152
+ return result
153
+ }
154
+
155
+ // ============================================================================
156
+ // Exports
157
+ // ============================================================================
158
+
159
+ export { toWords }
@@ -1,30 +1,17 @@
1
1
  /**
2
- * Arabic language converter.
2
+ * Converts a numeric value to Arabic words.
3
3
  *
4
- * Supports:
5
- * - Gender agreement (masculine/feminine forms)
6
- * - Complex pluralization (singular/dual/plural)
7
- * - Traditional Arabic number naming conventions
8
- * - Right-to-left text orientation
4
+ * @param {number | string | bigint} value - The numeric value to convert
5
+ * @param {Object} [options] - Optional configuration
6
+ * @param {('masculine'|'feminine')} [options.gender='masculine'] - Grammatical gender
7
+ * @param {string} [options.negativeWord] - Custom word for negative numbers
8
+ * @returns {string} The number in Arabic words
9
+ *
10
+ * @example
11
+ * toWords(1) // 'واحد'
12
+ * toWords(1, {gender: 'feminine'}) // 'واحدة'
9
13
  */
10
- export class Arabic extends AbstractLanguage {
11
- constructor(options?: {});
12
- tensWords: string[];
13
- hundredsWords: string[];
14
- scaleWords: string[];
15
- scaleAppendedWords: string[];
16
- scalePluralWords: string[];
17
- dualWords: string[];
18
- dualAppendedWords: string[];
19
- ones: {
20
- masculine: string[];
21
- feminine: string[];
22
- };
23
- /** Selects masculine or feminine number forms based on options. */
24
- get selectedOnes(): string[];
25
- /** Converts a 3-digit segment (0-999) to Arabic words with gender/plural rules. */
26
- segmentToWords(groupNumber: any, groupLevel: any, fullNumber: any): string;
27
- /** Converts integer part to Arabic words by processing 3-digit groups. */
28
- integerToWords(integerPart: any): string;
29
- }
30
- import { AbstractLanguage } from '../classes/abstract-language.js';
14
+ export function toWords(value: number | string | bigint, options?: {
15
+ gender?: "masculine" | "feminine" | undefined;
16
+ negativeWord?: string | undefined;
17
+ }): string;