@keenthemes/ktui 1.2.1 → 1.2.3

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 (304) hide show
  1. package/dist/ktui.js +2216 -845
  2. package/dist/ktui.min.js +1 -1
  3. package/dist/ktui.min.js.map +1 -1
  4. package/dist/styles.css +24 -197
  5. package/lib/cjs/components/carousel/carousel.d.ts +102 -0
  6. package/lib/cjs/components/carousel/carousel.d.ts.map +1 -0
  7. package/lib/cjs/components/carousel/carousel.js +769 -0
  8. package/lib/cjs/components/carousel/carousel.js.map +1 -0
  9. package/lib/cjs/components/carousel/index.d.ts +7 -0
  10. package/lib/cjs/components/carousel/index.d.ts.map +1 -0
  11. package/lib/cjs/components/carousel/index.js +10 -0
  12. package/lib/cjs/components/carousel/index.js.map +1 -0
  13. package/lib/cjs/components/carousel/types.d.ts +36 -0
  14. package/lib/cjs/components/carousel/types.d.ts.map +1 -0
  15. package/lib/cjs/components/carousel/types.js +7 -0
  16. package/lib/cjs/components/carousel/types.js.map +1 -0
  17. package/lib/cjs/components/component.d.ts +3 -3
  18. package/lib/cjs/components/component.d.ts.map +1 -1
  19. package/lib/cjs/components/component.js +9 -1
  20. package/lib/cjs/components/component.js.map +1 -1
  21. package/lib/cjs/components/datatable/datatable-checkbox.d.ts +1 -1
  22. package/lib/cjs/components/datatable/datatable-checkbox.d.ts.map +1 -1
  23. package/lib/cjs/components/datatable/datatable-checkbox.js +1 -1
  24. package/lib/cjs/components/datatable/datatable-checkbox.js.map +1 -1
  25. package/lib/cjs/components/datatable/datatable-sort.d.ts +1 -1
  26. package/lib/cjs/components/datatable/datatable-sort.d.ts.map +1 -1
  27. package/lib/cjs/components/datatable/datatable-sort.js.map +1 -1
  28. package/lib/cjs/components/datatable/datatable.d.ts +2 -0
  29. package/lib/cjs/components/datatable/datatable.d.ts.map +1 -1
  30. package/lib/cjs/components/datatable/datatable.js +30 -17
  31. package/lib/cjs/components/datatable/datatable.js.map +1 -1
  32. package/lib/cjs/components/datatable/types.d.ts +2 -1
  33. package/lib/cjs/components/datatable/types.d.ts.map +1 -1
  34. package/lib/cjs/components/drawer/drawer.d.ts.map +1 -1
  35. package/lib/cjs/components/drawer/drawer.js +3 -16
  36. package/lib/cjs/components/drawer/drawer.js.map +1 -1
  37. package/lib/cjs/components/dropdown/dropdown.d.ts +1 -1
  38. package/lib/cjs/components/dropdown/dropdown.d.ts.map +1 -1
  39. package/lib/cjs/components/dropdown/dropdown.js +2 -3
  40. package/lib/cjs/components/dropdown/dropdown.js.map +1 -1
  41. package/lib/cjs/components/modal/modal.d.ts.map +1 -1
  42. package/lib/cjs/components/modal/modal.js +19 -13
  43. package/lib/cjs/components/modal/modal.js.map +1 -1
  44. package/lib/cjs/components/pin-input/index.d.ts +3 -0
  45. package/lib/cjs/components/pin-input/index.d.ts.map +1 -0
  46. package/lib/cjs/components/pin-input/index.js +6 -0
  47. package/lib/cjs/components/pin-input/index.js.map +1 -0
  48. package/lib/cjs/components/pin-input/pin-input.d.ts +56 -0
  49. package/lib/cjs/components/pin-input/pin-input.d.ts.map +1 -0
  50. package/lib/cjs/components/pin-input/pin-input.js +455 -0
  51. package/lib/cjs/components/pin-input/pin-input.js.map +1 -0
  52. package/lib/cjs/components/pin-input/types.d.ts +41 -0
  53. package/lib/cjs/components/pin-input/types.d.ts.map +1 -0
  54. package/lib/cjs/components/pin-input/types.js +6 -0
  55. package/lib/cjs/components/pin-input/types.js.map +1 -0
  56. package/lib/cjs/components/rating/rating.d.ts.map +1 -1
  57. package/lib/cjs/components/rating/rating.js.map +1 -1
  58. package/lib/cjs/components/select/combobox.d.ts.map +1 -1
  59. package/lib/cjs/components/select/combobox.js +25 -15
  60. package/lib/cjs/components/select/combobox.js.map +1 -1
  61. package/lib/cjs/components/select/config.d.ts +2 -2
  62. package/lib/cjs/components/select/config.d.ts.map +1 -1
  63. package/lib/cjs/components/select/config.js +10 -9
  64. package/lib/cjs/components/select/config.js.map +1 -1
  65. package/lib/cjs/components/select/dropdown.js.map +1 -1
  66. package/lib/cjs/components/select/option.d.ts +2 -1
  67. package/lib/cjs/components/select/option.d.ts.map +1 -1
  68. package/lib/cjs/components/select/option.js +9 -3
  69. package/lib/cjs/components/select/option.js.map +1 -1
  70. package/lib/cjs/components/select/remote.d.ts +1 -0
  71. package/lib/cjs/components/select/remote.d.ts.map +1 -1
  72. package/lib/cjs/components/select/remote.js +21 -14
  73. package/lib/cjs/components/select/remote.js.map +1 -1
  74. package/lib/cjs/components/select/search.d.ts +1 -1
  75. package/lib/cjs/components/select/search.d.ts.map +1 -1
  76. package/lib/cjs/components/select/search.js +34 -25
  77. package/lib/cjs/components/select/search.js.map +1 -1
  78. package/lib/cjs/components/select/select.d.ts +5 -3
  79. package/lib/cjs/components/select/select.d.ts.map +1 -1
  80. package/lib/cjs/components/select/select.js +31 -31
  81. package/lib/cjs/components/select/select.js.map +1 -1
  82. package/lib/cjs/components/select/tags.d.ts.map +1 -1
  83. package/lib/cjs/components/select/tags.js +22 -13
  84. package/lib/cjs/components/select/tags.js.map +1 -1
  85. package/lib/cjs/components/select/templates.d.ts.map +1 -1
  86. package/lib/cjs/components/select/templates.js +4 -4
  87. package/lib/cjs/components/select/templates.js.map +1 -1
  88. package/lib/cjs/components/select/types.d.ts +1 -1
  89. package/lib/cjs/components/select/types.d.ts.map +1 -1
  90. package/lib/cjs/components/select/utils.d.ts +4 -4
  91. package/lib/cjs/components/select/utils.d.ts.map +1 -1
  92. package/lib/cjs/components/select/utils.js +2 -3
  93. package/lib/cjs/components/select/utils.js.map +1 -1
  94. package/lib/cjs/components/sticky/sticky.d.ts +1 -1
  95. package/lib/cjs/components/sticky/sticky.d.ts.map +1 -1
  96. package/lib/cjs/components/sticky/sticky.js +13 -13
  97. package/lib/cjs/components/sticky/sticky.js.map +1 -1
  98. package/lib/cjs/components/theme-switch/theme-switch.d.ts +3 -0
  99. package/lib/cjs/components/theme-switch/theme-switch.d.ts.map +1 -1
  100. package/lib/cjs/components/theme-switch/theme-switch.js +17 -4
  101. package/lib/cjs/components/theme-switch/theme-switch.js.map +1 -1
  102. package/lib/cjs/components/toast/toast.d.ts.map +1 -1
  103. package/lib/cjs/components/toast/toast.js +17 -9
  104. package/lib/cjs/components/toast/toast.js.map +1 -1
  105. package/lib/cjs/components/toast/types.d.ts +3 -0
  106. package/lib/cjs/components/toast/types.d.ts.map +1 -1
  107. package/lib/cjs/components/toggle/toggle.d.ts +2 -0
  108. package/lib/cjs/components/toggle/toggle.d.ts.map +1 -1
  109. package/lib/cjs/components/toggle/toggle.js +11 -2
  110. package/lib/cjs/components/toggle/toggle.js.map +1 -1
  111. package/lib/cjs/components/toggle-password/toggle-password.d.ts.map +1 -1
  112. package/lib/cjs/components/toggle-password/toggle-password.js.map +1 -1
  113. package/lib/cjs/helpers/dom.d.ts +4 -4
  114. package/lib/cjs/helpers/dom.d.ts.map +1 -1
  115. package/lib/cjs/helpers/dom.js +8 -10
  116. package/lib/cjs/helpers/dom.js.map +1 -1
  117. package/lib/cjs/helpers/event-handler.d.ts +1 -1
  118. package/lib/cjs/helpers/event-handler.d.ts.map +1 -1
  119. package/lib/cjs/helpers/event-handler.js +3 -1
  120. package/lib/cjs/helpers/event-handler.js.map +1 -1
  121. package/lib/cjs/helpers/utils.d.ts +1 -1
  122. package/lib/cjs/helpers/utils.d.ts.map +1 -1
  123. package/lib/cjs/helpers/utils.js +4 -1
  124. package/lib/cjs/helpers/utils.js.map +1 -1
  125. package/lib/cjs/index.d.ts +8 -0
  126. package/lib/cjs/index.d.ts.map +1 -1
  127. package/lib/cjs/index.js +9 -1
  128. package/lib/cjs/index.js.map +1 -1
  129. package/lib/cjs/types.d.ts +1 -1
  130. package/lib/cjs/types.d.ts.map +1 -1
  131. package/lib/esm/components/carousel/carousel.d.ts +102 -0
  132. package/lib/esm/components/carousel/carousel.d.ts.map +1 -0
  133. package/lib/esm/components/carousel/carousel.js +766 -0
  134. package/lib/esm/components/carousel/carousel.js.map +1 -0
  135. package/lib/esm/components/carousel/index.d.ts +7 -0
  136. package/lib/esm/components/carousel/index.d.ts.map +1 -0
  137. package/lib/esm/components/carousel/index.js +6 -0
  138. package/lib/esm/components/carousel/index.js.map +1 -0
  139. package/lib/esm/components/carousel/types.d.ts +36 -0
  140. package/lib/esm/components/carousel/types.d.ts.map +1 -0
  141. package/lib/esm/components/carousel/types.js +6 -0
  142. package/lib/esm/components/carousel/types.js.map +1 -0
  143. package/lib/esm/components/component.d.ts +3 -3
  144. package/lib/esm/components/component.d.ts.map +1 -1
  145. package/lib/esm/components/component.js +9 -1
  146. package/lib/esm/components/component.js.map +1 -1
  147. package/lib/esm/components/datatable/datatable-checkbox.d.ts +1 -1
  148. package/lib/esm/components/datatable/datatable-checkbox.d.ts.map +1 -1
  149. package/lib/esm/components/datatable/datatable-checkbox.js +1 -1
  150. package/lib/esm/components/datatable/datatable-checkbox.js.map +1 -1
  151. package/lib/esm/components/datatable/datatable-sort.d.ts +1 -1
  152. package/lib/esm/components/datatable/datatable-sort.d.ts.map +1 -1
  153. package/lib/esm/components/datatable/datatable-sort.js.map +1 -1
  154. package/lib/esm/components/datatable/datatable.d.ts +2 -0
  155. package/lib/esm/components/datatable/datatable.d.ts.map +1 -1
  156. package/lib/esm/components/datatable/datatable.js +30 -17
  157. package/lib/esm/components/datatable/datatable.js.map +1 -1
  158. package/lib/esm/components/datatable/types.d.ts +2 -1
  159. package/lib/esm/components/datatable/types.d.ts.map +1 -1
  160. package/lib/esm/components/drawer/drawer.d.ts.map +1 -1
  161. package/lib/esm/components/drawer/drawer.js +3 -16
  162. package/lib/esm/components/drawer/drawer.js.map +1 -1
  163. package/lib/esm/components/dropdown/dropdown.d.ts +1 -1
  164. package/lib/esm/components/dropdown/dropdown.d.ts.map +1 -1
  165. package/lib/esm/components/dropdown/dropdown.js +2 -3
  166. package/lib/esm/components/dropdown/dropdown.js.map +1 -1
  167. package/lib/esm/components/modal/modal.d.ts.map +1 -1
  168. package/lib/esm/components/modal/modal.js +19 -13
  169. package/lib/esm/components/modal/modal.js.map +1 -1
  170. package/lib/esm/components/pin-input/index.d.ts +3 -0
  171. package/lib/esm/components/pin-input/index.d.ts.map +1 -0
  172. package/lib/esm/components/pin-input/index.js +2 -0
  173. package/lib/esm/components/pin-input/index.js.map +1 -0
  174. package/lib/esm/components/pin-input/pin-input.d.ts +56 -0
  175. package/lib/esm/components/pin-input/pin-input.d.ts.map +1 -0
  176. package/lib/esm/components/pin-input/pin-input.js +452 -0
  177. package/lib/esm/components/pin-input/pin-input.js.map +1 -0
  178. package/lib/esm/components/pin-input/types.d.ts +41 -0
  179. package/lib/esm/components/pin-input/types.d.ts.map +1 -0
  180. package/lib/esm/components/pin-input/types.js +5 -0
  181. package/lib/esm/components/pin-input/types.js.map +1 -0
  182. package/lib/esm/components/rating/rating.d.ts.map +1 -1
  183. package/lib/esm/components/rating/rating.js.map +1 -1
  184. package/lib/esm/components/select/combobox.d.ts.map +1 -1
  185. package/lib/esm/components/select/combobox.js +25 -15
  186. package/lib/esm/components/select/combobox.js.map +1 -1
  187. package/lib/esm/components/select/config.d.ts +2 -2
  188. package/lib/esm/components/select/config.d.ts.map +1 -1
  189. package/lib/esm/components/select/config.js +10 -9
  190. package/lib/esm/components/select/config.js.map +1 -1
  191. package/lib/esm/components/select/dropdown.js.map +1 -1
  192. package/lib/esm/components/select/option.d.ts +2 -1
  193. package/lib/esm/components/select/option.d.ts.map +1 -1
  194. package/lib/esm/components/select/option.js +9 -3
  195. package/lib/esm/components/select/option.js.map +1 -1
  196. package/lib/esm/components/select/remote.d.ts +1 -0
  197. package/lib/esm/components/select/remote.d.ts.map +1 -1
  198. package/lib/esm/components/select/remote.js +21 -14
  199. package/lib/esm/components/select/remote.js.map +1 -1
  200. package/lib/esm/components/select/search.d.ts +1 -1
  201. package/lib/esm/components/select/search.d.ts.map +1 -1
  202. package/lib/esm/components/select/search.js +34 -25
  203. package/lib/esm/components/select/search.js.map +1 -1
  204. package/lib/esm/components/select/select.d.ts +5 -3
  205. package/lib/esm/components/select/select.d.ts.map +1 -1
  206. package/lib/esm/components/select/select.js +31 -31
  207. package/lib/esm/components/select/select.js.map +1 -1
  208. package/lib/esm/components/select/tags.d.ts.map +1 -1
  209. package/lib/esm/components/select/tags.js +22 -13
  210. package/lib/esm/components/select/tags.js.map +1 -1
  211. package/lib/esm/components/select/templates.d.ts.map +1 -1
  212. package/lib/esm/components/select/templates.js +4 -4
  213. package/lib/esm/components/select/templates.js.map +1 -1
  214. package/lib/esm/components/select/types.d.ts +1 -1
  215. package/lib/esm/components/select/types.d.ts.map +1 -1
  216. package/lib/esm/components/select/utils.d.ts +4 -4
  217. package/lib/esm/components/select/utils.d.ts.map +1 -1
  218. package/lib/esm/components/select/utils.js +2 -3
  219. package/lib/esm/components/select/utils.js.map +1 -1
  220. package/lib/esm/components/sticky/sticky.d.ts +1 -1
  221. package/lib/esm/components/sticky/sticky.d.ts.map +1 -1
  222. package/lib/esm/components/sticky/sticky.js +13 -13
  223. package/lib/esm/components/sticky/sticky.js.map +1 -1
  224. package/lib/esm/components/theme-switch/theme-switch.d.ts +3 -0
  225. package/lib/esm/components/theme-switch/theme-switch.d.ts.map +1 -1
  226. package/lib/esm/components/theme-switch/theme-switch.js +17 -4
  227. package/lib/esm/components/theme-switch/theme-switch.js.map +1 -1
  228. package/lib/esm/components/toast/toast.d.ts.map +1 -1
  229. package/lib/esm/components/toast/toast.js +17 -9
  230. package/lib/esm/components/toast/toast.js.map +1 -1
  231. package/lib/esm/components/toast/types.d.ts +3 -0
  232. package/lib/esm/components/toast/types.d.ts.map +1 -1
  233. package/lib/esm/components/toggle/toggle.d.ts +2 -0
  234. package/lib/esm/components/toggle/toggle.d.ts.map +1 -1
  235. package/lib/esm/components/toggle/toggle.js +11 -2
  236. package/lib/esm/components/toggle/toggle.js.map +1 -1
  237. package/lib/esm/components/toggle-password/toggle-password.d.ts.map +1 -1
  238. package/lib/esm/components/toggle-password/toggle-password.js.map +1 -1
  239. package/lib/esm/helpers/dom.d.ts +4 -4
  240. package/lib/esm/helpers/dom.d.ts.map +1 -1
  241. package/lib/esm/helpers/dom.js +8 -10
  242. package/lib/esm/helpers/dom.js.map +1 -1
  243. package/lib/esm/helpers/event-handler.d.ts +1 -1
  244. package/lib/esm/helpers/event-handler.d.ts.map +1 -1
  245. package/lib/esm/helpers/event-handler.js +3 -1
  246. package/lib/esm/helpers/event-handler.js.map +1 -1
  247. package/lib/esm/helpers/utils.d.ts +1 -1
  248. package/lib/esm/helpers/utils.d.ts.map +1 -1
  249. package/lib/esm/helpers/utils.js +4 -1
  250. package/lib/esm/helpers/utils.js.map +1 -1
  251. package/lib/esm/index.d.ts +8 -0
  252. package/lib/esm/index.d.ts.map +1 -1
  253. package/lib/esm/index.js +6 -0
  254. package/lib/esm/index.js.map +1 -1
  255. package/lib/esm/types.d.ts +1 -1
  256. package/lib/esm/types.d.ts.map +1 -1
  257. package/package.json +5 -2
  258. package/src/components/carousel/__tests__/carousel.test.ts +326 -0
  259. package/src/components/carousel/carousel.css +42 -0
  260. package/src/components/carousel/carousel.ts +847 -0
  261. package/src/components/carousel/index.ts +11 -0
  262. package/src/components/carousel/types.ts +38 -0
  263. package/src/components/clipboard/__tests__/clipboard.test.ts +4 -4
  264. package/src/components/component.ts +15 -5
  265. package/src/components/datatable/__tests__/currency-sort.test.ts +4 -3
  266. package/src/components/datatable/__tests__/pagination-reset.test.ts +7 -4
  267. package/src/components/datatable/__tests__/setup.ts +1 -1
  268. package/src/components/datatable/datatable-checkbox.ts +6 -4
  269. package/src/components/datatable/datatable-sort.ts +27 -7
  270. package/src/components/datatable/datatable.ts +64 -37
  271. package/src/components/datatable/types.ts +3 -1
  272. package/src/components/drawer/drawer.ts +3 -18
  273. package/src/components/dropdown/dropdown.ts +2 -3
  274. package/src/components/modal/modal.ts +22 -14
  275. package/src/components/pin-input/__tests__/pin-input.test.ts +928 -0
  276. package/src/components/pin-input/index.ts +6 -0
  277. package/src/components/pin-input/pin-input.ts +499 -0
  278. package/src/components/pin-input/types.ts +45 -0
  279. package/src/components/rating/rating.ts +0 -1
  280. package/src/components/repeater/__tests__/repeater.test.ts +5 -5
  281. package/src/components/select/__tests__/ux-behaviors.test.ts +4 -3
  282. package/src/components/select/combobox.ts +23 -16
  283. package/src/components/select/config.ts +15 -14
  284. package/src/components/select/dropdown.ts +1 -1
  285. package/src/components/select/option.ts +14 -4
  286. package/src/components/select/remote.ts +68 -56
  287. package/src/components/select/search.ts +30 -27
  288. package/src/components/select/select.ts +41 -37
  289. package/src/components/select/tags.ts +14 -8
  290. package/src/components/select/templates.ts +11 -6
  291. package/src/components/select/types.ts +1 -1
  292. package/src/components/select/utils.ts +7 -9
  293. package/src/components/sticky/sticky.ts +2 -2
  294. package/src/components/theme-switch/theme-switch.ts +22 -4
  295. package/src/components/toast/toast.ts +34 -21
  296. package/src/components/toast/types.ts +5 -1
  297. package/src/components/toggle/toggle.ts +12 -2
  298. package/src/components/toggle-password/toggle-password.ts +0 -1
  299. package/src/helpers/dom.ts +14 -17
  300. package/src/helpers/event-handler.ts +5 -6
  301. package/src/helpers/utils.ts +5 -2
  302. package/src/index.ts +18 -0
  303. package/src/types.ts +1 -1
  304. package/styles.css +1 -0
@@ -0,0 +1,326 @@
1
+ /**
2
+ * Tests for KTCarousel component
3
+ */
4
+ import { afterEach, describe, expect, it, vi } from 'vitest';
5
+ import { KTCarousel } from '../carousel';
6
+
7
+ function buildCarouselHtml(options?: {
8
+ infinite?: boolean;
9
+ autoplay?: boolean;
10
+ lazy?: boolean;
11
+ snap?: boolean;
12
+ draggable?: boolean;
13
+ }) {
14
+ const root = document.createElement('div');
15
+ root.setAttribute('data-kt-carousel', 'true');
16
+ if (options?.lazy) root.setAttribute('data-kt-carousel-lazy', 'true');
17
+ if (options?.infinite)
18
+ root.setAttribute('data-kt-carousel-infinite-loop', 'true');
19
+ if (options?.autoplay) root.setAttribute('data-kt-carousel-autoplay', 'true');
20
+ if (options?.snap) root.setAttribute('data-kt-carousel-snap', 'true');
21
+ if (options?.draggable)
22
+ root.setAttribute('data-kt-carousel-draggable', 'true');
23
+
24
+ const viewport = document.createElement('div');
25
+ viewport.setAttribute('data-kt-carousel-viewport', 'true');
26
+ viewport.style.overflow = 'auto';
27
+ viewport.style.width = '200px';
28
+
29
+ for (let i = 0; i < 3; i++) {
30
+ const slide = document.createElement('div');
31
+ slide.setAttribute('data-kt-carousel-item', 'true');
32
+ slide.style.width = '200px';
33
+ slide.style.flexShrink = '0';
34
+ slide.textContent = `S${i}`;
35
+ viewport.appendChild(slide);
36
+ }
37
+
38
+ const next = document.createElement('button');
39
+ next.type = 'button';
40
+ next.setAttribute('data-kt-carousel-next', 'true');
41
+ const prev = document.createElement('button');
42
+ prev.type = 'button';
43
+ prev.setAttribute('data-kt-carousel-prev', 'true');
44
+
45
+ root.appendChild(viewport);
46
+ root.appendChild(prev);
47
+ root.appendChild(next);
48
+ document.body.appendChild(root);
49
+
50
+ const slides = Array.from(
51
+ viewport.querySelectorAll<HTMLElement>('[data-kt-carousel-item]'),
52
+ );
53
+ const gap = 200;
54
+ slides.forEach((slide, i) => {
55
+ Object.defineProperty(slide, 'offsetLeft', {
56
+ configurable: true,
57
+ get: () => i * gap,
58
+ });
59
+ Object.defineProperty(slide, 'offsetHeight', {
60
+ configurable: true,
61
+ get: () => 100,
62
+ });
63
+ });
64
+
65
+ let scrollLeft = 0;
66
+ Object.defineProperty(viewport, 'scrollLeft', {
67
+ configurable: true,
68
+ get: () => scrollLeft,
69
+ set: (v: number) => {
70
+ scrollLeft = v;
71
+ },
72
+ });
73
+ Object.defineProperty(viewport, 'scrollWidth', {
74
+ configurable: true,
75
+ get: () => slides.length * gap,
76
+ });
77
+ Object.defineProperty(viewport, 'clientWidth', {
78
+ configurable: true,
79
+ get: () => gap,
80
+ });
81
+
82
+ const vpScreenLeft = 10;
83
+ vi.spyOn(viewport, 'getBoundingClientRect').mockReturnValue({
84
+ left: vpScreenLeft,
85
+ top: 0,
86
+ width: gap,
87
+ height: 100,
88
+ right: vpScreenLeft + gap,
89
+ bottom: 100,
90
+ x: vpScreenLeft,
91
+ y: 0,
92
+ toJSON: () => ({}),
93
+ } as DOMRect);
94
+
95
+ slides.forEach((slide, i) => {
96
+ vi.spyOn(slide, 'getBoundingClientRect').mockImplementation(() => {
97
+ const visualLeft = vpScreenLeft + i * gap - scrollLeft;
98
+ return {
99
+ left: visualLeft,
100
+ top: 0,
101
+ width: gap,
102
+ height: 100,
103
+ right: visualLeft + gap,
104
+ bottom: 100,
105
+ x: visualLeft,
106
+ y: 0,
107
+ toJSON: () => ({}),
108
+ } as DOMRect;
109
+ });
110
+ });
111
+
112
+ viewport.scrollTo = ((opts: ScrollToOptions) => {
113
+ if (typeof opts.left === 'number') {
114
+ scrollLeft = opts.left;
115
+ }
116
+ }) as typeof viewport.scrollTo;
117
+
118
+ return {
119
+ root,
120
+ viewport,
121
+ slides,
122
+ restoreScrollIntoView: () => {
123
+ /* no-op; mocks cleared in afterEach */
124
+ },
125
+ };
126
+ }
127
+
128
+ describe('KTCarousel', () => {
129
+ afterEach(() => {
130
+ document.body.innerHTML = '';
131
+ vi.useRealTimers();
132
+ vi.restoreAllMocks();
133
+ });
134
+
135
+ it('goTo sets index after navigation settles', async () => {
136
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
137
+ try {
138
+ const c = new KTCarousel(root);
139
+ expect(c.getSlideCount()).toBe(3);
140
+ c.goTo(2);
141
+ await Promise.resolve();
142
+ expect(c.getIndex()).toBe(2);
143
+ } finally {
144
+ restoreScrollIntoView();
145
+ }
146
+ });
147
+
148
+ it('next advances and prev goes back', async () => {
149
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
150
+ try {
151
+ const c = new KTCarousel(root);
152
+ c.next(true);
153
+ await Promise.resolve();
154
+ expect(c.getIndex()).toBe(1);
155
+ c.prev(true);
156
+ await Promise.resolve();
157
+ expect(c.getIndex()).toBe(0);
158
+ } finally {
159
+ restoreScrollIntoView();
160
+ }
161
+ });
162
+
163
+ it('next at last without infinite does nothing', async () => {
164
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
165
+ try {
166
+ const c = new KTCarousel(root);
167
+ c.goTo(2);
168
+ await Promise.resolve();
169
+ c.next(true);
170
+ await Promise.resolve();
171
+ expect(c.getIndex()).toBe(2);
172
+ } finally {
173
+ restoreScrollIntoView();
174
+ }
175
+ });
176
+
177
+ it('next at last with infinite wraps to 0', async () => {
178
+ const { root, restoreScrollIntoView } = buildCarouselHtml({
179
+ infinite: true,
180
+ });
181
+ try {
182
+ const c = new KTCarousel(root);
183
+ c.goTo(2);
184
+ await Promise.resolve();
185
+ c.next(true);
186
+ await Promise.resolve();
187
+ expect(c.getIndex()).toBe(0);
188
+ } finally {
189
+ restoreScrollIntoView();
190
+ }
191
+ });
192
+
193
+ it('dispose clears instance', () => {
194
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
195
+ try {
196
+ const c = new KTCarousel(root);
197
+ c.dispose();
198
+ expect(KTCarousel.getInstance(root)).toBeNull();
199
+ } finally {
200
+ restoreScrollIntoView();
201
+ }
202
+ });
203
+
204
+ it('dispose is idempotent', () => {
205
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
206
+ try {
207
+ const c = new KTCarousel(root);
208
+ c.dispose();
209
+ c.dispose();
210
+ expect(KTCarousel.getInstance(root)).toBeNull();
211
+ } finally {
212
+ restoreScrollIntoView();
213
+ }
214
+ });
215
+
216
+ it('dispatches change event on next', async () => {
217
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
218
+ try {
219
+ const c = new KTCarousel(root);
220
+ const spy = vi.fn();
221
+ root.addEventListener('kt.carousel.change', spy);
222
+ c.next(true);
223
+ await Promise.resolve();
224
+ expect(spy).toHaveBeenCalled();
225
+ } finally {
226
+ restoreScrollIntoView();
227
+ }
228
+ });
229
+
230
+ it('createInstances skips lazy roots', () => {
231
+ const { root, restoreScrollIntoView } = buildCarouselHtml({ lazy: true });
232
+ try {
233
+ KTCarousel.createInstances();
234
+ expect(KTCarousel.getInstance(root)).toBeNull();
235
+ } finally {
236
+ restoreScrollIntoView();
237
+ }
238
+ });
239
+
240
+ it('prefers-reduced-motion disables autoplay timer', () => {
241
+ vi.useFakeTimers();
242
+ const mm = vi.fn().mockImplementation((q: string) => ({
243
+ matches: q.includes('prefers-reduced-motion'),
244
+ media: q,
245
+ onchange: null,
246
+ addListener: vi.fn(),
247
+ removeListener: vi.fn(),
248
+ addEventListener: vi.fn(),
249
+ removeEventListener: vi.fn(),
250
+ dispatchEvent: vi.fn(),
251
+ }));
252
+ vi.stubGlobal('matchMedia', mm);
253
+
254
+ const { root, restoreScrollIntoView } = buildCarouselHtml({
255
+ autoplay: true,
256
+ });
257
+ try {
258
+ const c = new KTCarousel(root);
259
+ vi.advanceTimersByTime(20000);
260
+ expect(c.getIndex()).toBe(0);
261
+ } finally {
262
+ restoreScrollIntoView();
263
+ }
264
+ });
265
+
266
+ it('getOrCreateInstance returns same instance', () => {
267
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
268
+ try {
269
+ const a = KTCarousel.getOrCreateInstance(root);
270
+ const b = KTCarousel.getOrCreateInstance(root);
271
+ expect(a).toBe(b);
272
+ } finally {
273
+ restoreScrollIntoView();
274
+ }
275
+ });
276
+
277
+ it('showScrollbar config sets data-kt-carousel-show-scrollbar on root', () => {
278
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
279
+ try {
280
+ new KTCarousel(root, { showScrollbar: true });
281
+ expect(root.getAttribute('data-kt-carousel-show-scrollbar')).toBe('true');
282
+ } finally {
283
+ restoreScrollIntoView();
284
+ }
285
+ });
286
+
287
+ it('default showScrollbar removes data-kt-carousel-show-scrollbar', () => {
288
+ const { root, restoreScrollIntoView } = buildCarouselHtml();
289
+ root.setAttribute('data-kt-carousel-show-scrollbar', 'true');
290
+ try {
291
+ new KTCarousel(root, { showScrollbar: false });
292
+ expect(root.hasAttribute('data-kt-carousel-show-scrollbar')).toBe(false);
293
+ } finally {
294
+ restoreScrollIntoView();
295
+ }
296
+ });
297
+
298
+ it('snap mode does not attach pointer drag listeners', () => {
299
+ const { root, viewport, restoreScrollIntoView } = buildCarouselHtml({
300
+ snap: true,
301
+ draggable: true,
302
+ });
303
+ try {
304
+ const spy = vi.spyOn(viewport, 'addEventListener');
305
+ new KTCarousel(root);
306
+ const pointerDown = spy.mock.calls.some((c) => c[0] === 'pointerdown');
307
+ expect(pointerDown).toBe(false);
308
+ } finally {
309
+ restoreScrollIntoView();
310
+ }
311
+ });
312
+
313
+ it('draggable without snap attaches pointerdown', () => {
314
+ const { root, viewport, restoreScrollIntoView } = buildCarouselHtml({
315
+ draggable: true,
316
+ });
317
+ try {
318
+ const spy = vi.spyOn(viewport, 'addEventListener');
319
+ new KTCarousel(root);
320
+ const pointerDown = spy.mock.calls.some((c) => c[0] === 'pointerdown');
321
+ expect(pointerDown).toBe(true);
322
+ } finally {
323
+ restoreScrollIntoView();
324
+ }
325
+ });
326
+ });
@@ -0,0 +1,42 @@
1
+ /**
2
+ * KTCarousel — by default hides native scrollbars on the viewport and thumbnail
3
+ * strips; scrolling still works via trackpad, wheel, drag, and scrollTo.
4
+ * Set `data-kt-carousel-show-scrollbar="true"` on the root to keep scrollbars visible.
5
+ */
6
+ @layer components {
7
+ [data-kt-carousel]:not([data-kt-carousel-show-scrollbar="true"])
8
+ [data-kt-carousel-viewport],
9
+ [data-kt-carousel]:not([data-kt-carousel-show-scrollbar="true"])
10
+ [data-kt-carousel-thumbnails] {
11
+ scrollbar-width: none;
12
+ -ms-overflow-style: none;
13
+ }
14
+
15
+ [data-kt-carousel]:not([data-kt-carousel-show-scrollbar="true"])
16
+ [data-kt-carousel-viewport]::-webkit-scrollbar,
17
+ [data-kt-carousel]:not([data-kt-carousel-show-scrollbar="true"])
18
+ [data-kt-carousel-thumbnails]::-webkit-scrollbar {
19
+ display: none;
20
+ }
21
+
22
+ [data-kt-carousel-thumbnail]:not(:disabled),
23
+ [data-kt-carousel-pagination-item]:not(:disabled),
24
+ [data-kt-carousel-prev]:not(:disabled),
25
+ [data-kt-carousel-next]:not(:disabled) {
26
+ cursor: pointer;
27
+ }
28
+
29
+ /* Smooth active-state visuals when `data-kt-carousel-thumbnail-active` toggles */
30
+ [data-kt-carousel-thumbnail] {
31
+ transition-property: border-color, background-color, color, box-shadow, opacity;
32
+ transition-duration: 0.2s;
33
+ transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
34
+ }
35
+
36
+ @media (prefers-reduced-motion: reduce) {
37
+ [data-kt-carousel-thumbnail] {
38
+ transition-duration: 0.01ms;
39
+ transition-property: none;
40
+ }
41
+ }
42
+ }