@windstream/react-shared-components 0.1.92 → 0.1.94

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 (293) hide show
  1. package/README.md +635 -635
  2. package/dist/contentful/index.d.ts +1 -0
  3. package/dist/contentful/index.esm.js +2 -2
  4. package/dist/contentful/index.esm.js.map +1 -1
  5. package/dist/contentful/index.js +3 -3
  6. package/dist/contentful/index.js.map +1 -1
  7. package/dist/core.d.ts +5 -5
  8. package/dist/index.d.ts +2 -2
  9. package/dist/index.esm.js +1 -1
  10. package/dist/index.esm.js.map +1 -1
  11. package/dist/index.js +1 -1
  12. package/dist/index.js.map +1 -1
  13. package/dist/styles.css +1 -1
  14. package/dist/utils/index.esm.js +1 -1
  15. package/dist/utils/index.js +1 -1
  16. package/package.json +191 -185
  17. package/src/components/accordion/Accordion.stories.tsx +230 -230
  18. package/src/components/accordion/index.test.tsx +270 -0
  19. package/src/components/accordion/index.tsx +70 -70
  20. package/src/components/accordion/types.ts +12 -12
  21. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  22. package/src/components/alert-card/index.test.tsx +152 -0
  23. package/src/components/alert-card/index.tsx +41 -41
  24. package/src/components/alert-card/types.ts +13 -13
  25. package/src/components/animation-wrapper/index.test.tsx +424 -0
  26. package/src/components/animation-wrapper/index.tsx +129 -129
  27. package/src/components/animation-wrapper/types.ts +11 -11
  28. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  29. package/src/components/brand-button/helpers.ts +35 -35
  30. package/src/components/brand-button/index.test.tsx +292 -0
  31. package/src/components/brand-button/index.tsx +120 -120
  32. package/src/components/brand-button/types.ts +38 -38
  33. package/src/components/button/Button.stories.tsx +108 -108
  34. package/src/components/button/index.test.tsx +91 -0
  35. package/src/components/button/index.tsx +27 -27
  36. package/src/components/button/types.ts +14 -14
  37. package/src/components/call-button/CallButton.stories.tsx +324 -324
  38. package/src/components/call-button/index.test.tsx +260 -0
  39. package/src/components/call-button/index.tsx +106 -106
  40. package/src/components/call-button/types.ts +16 -16
  41. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  42. package/src/components/checkbox/index.test.tsx +252 -0
  43. package/src/components/checkbox/index.tsx +197 -197
  44. package/src/components/checkbox/types.ts +27 -27
  45. package/src/components/checklist/Checklist.stories.tsx +150 -150
  46. package/src/components/checklist/index.test.tsx +231 -0
  47. package/src/components/checklist/index.tsx +96 -61
  48. package/src/components/checklist/types.ts +23 -17
  49. package/src/components/collapse/Collapse.stories.tsx +255 -255
  50. package/src/components/collapse/index.test.tsx +277 -0
  51. package/src/components/collapse/index.tsx +47 -46
  52. package/src/components/collapse/types.ts +6 -6
  53. package/src/components/divider/Divider.stories.tsx +205 -205
  54. package/src/components/divider/index.test.tsx +53 -0
  55. package/src/components/divider/index.tsx +22 -22
  56. package/src/components/divider/type.ts +3 -3
  57. package/src/components/image/Image.stories.tsx +113 -113
  58. package/src/components/image/index.test.tsx +174 -0
  59. package/src/components/image/index.tsx +25 -25
  60. package/src/components/image/types.ts +40 -40
  61. package/src/components/input/Input.stories.tsx +325 -325
  62. package/src/components/input/index.test.tsx +348 -0
  63. package/src/components/input/index.tsx +177 -177
  64. package/src/components/input/types.ts +37 -37
  65. package/src/components/link/Link.stories.tsx +163 -163
  66. package/src/components/link/index.test.tsx +199 -0
  67. package/src/components/link/index.tsx +116 -116
  68. package/src/components/link/types.ts +25 -25
  69. package/src/components/list/List.stories.tsx +272 -272
  70. package/src/components/list/index.test.tsx +166 -0
  71. package/src/components/list/index.tsx +88 -88
  72. package/src/components/list/list-item/index.tsx +38 -38
  73. package/src/components/list/list-item/types.ts +13 -13
  74. package/src/components/list/types.ts +29 -29
  75. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  76. package/src/components/material-icon/constants.ts +99 -99
  77. package/src/components/material-icon/index.test.tsx +130 -0
  78. package/src/components/material-icon/index.tsx +47 -47
  79. package/src/components/material-icon/types.ts +31 -31
  80. package/src/components/modal/Modal.stories.tsx +171 -171
  81. package/src/components/modal/index.test.tsx +310 -0
  82. package/src/components/modal/index.tsx +164 -164
  83. package/src/components/modal/types.ts +24 -24
  84. package/src/components/next-image/index.test.tsx +406 -0
  85. package/src/components/next-image/index.tsx +74 -74
  86. package/src/components/next-image/types.ts +1 -1
  87. package/src/components/pagination/index.test.tsx +521 -0
  88. package/src/components/pagination/index.tsx +91 -91
  89. package/src/components/pagination/types.ts +6 -6
  90. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  91. package/src/components/radio-button/index.test.tsx +151 -0
  92. package/src/components/radio-button/index.tsx +75 -75
  93. package/src/components/radio-button/types.ts +21 -21
  94. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  95. package/src/components/see-more/index.test.tsx +96 -0
  96. package/src/components/see-more/index.tsx +44 -44
  97. package/src/components/see-more/types.ts +4 -4
  98. package/src/components/select/Select.stories.tsx +411 -411
  99. package/src/components/select/index.test.tsx +256 -0
  100. package/src/components/select/index.tsx +155 -155
  101. package/src/components/select/types.ts +36 -36
  102. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  103. package/src/components/select-plan-button/index.test.tsx +173 -0
  104. package/src/components/select-plan-button/index.tsx +63 -63
  105. package/src/components/select-plan-button/types.ts +17 -17
  106. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  107. package/src/components/skeleton/index.test.tsx +74 -0
  108. package/src/components/skeleton/index.tsx +61 -61
  109. package/src/components/skeleton/types.ts +4 -4
  110. package/src/components/spinner/Spinner.stories.tsx +335 -335
  111. package/src/components/spinner/index.test.tsx +76 -0
  112. package/src/components/spinner/index.tsx +44 -44
  113. package/src/components/spinner/types.ts +5 -5
  114. package/src/components/text/Text.stories.tsx +321 -321
  115. package/src/components/text/index.test.tsx +65 -0
  116. package/src/components/text/index.tsx +25 -25
  117. package/src/components/text/types.ts +45 -45
  118. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  119. package/src/components/tooltip/index.test.tsx +50 -0
  120. package/src/components/tooltip/index.tsx +74 -74
  121. package/src/components/tooltip/types.ts +7 -7
  122. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  123. package/src/components/view-cart-button/index.test.tsx +57 -0
  124. package/src/components/view-cart-button/index.tsx +42 -42
  125. package/src/components/view-cart-button/types.ts +5 -5
  126. package/src/contentful/blocks/accordion/Accordion.stories.mocks.tsx +128 -128
  127. package/src/contentful/blocks/accordion/Accordion.stories.tsx +98 -98
  128. package/src/contentful/blocks/accordion/index.test.tsx +218 -0
  129. package/src/contentful/blocks/accordion/index.tsx +114 -112
  130. package/src/contentful/blocks/accordion/types.ts +34 -34
  131. package/src/contentful/blocks/address-input-banner/index.test.tsx +132 -0
  132. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  133. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  134. package/src/contentful/blocks/anchored-bottom-banner/index.test.tsx +287 -0
  135. package/src/contentful/blocks/anchored-bottom-banner/index.tsx +181 -181
  136. package/src/contentful/blocks/anchored-bottom-banner/types.ts +13 -13
  137. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.mocks.tsx +144 -144
  138. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.tsx +157 -156
  139. package/src/contentful/blocks/blogs-grid/index.test.tsx +355 -0
  140. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  141. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  142. package/src/contentful/blocks/blogs-grid-base/index.test.tsx +274 -0
  143. package/src/contentful/blocks/blogs-grid-base/index.tsx +119 -119
  144. package/src/contentful/blocks/blogs-grid-base/types.ts +36 -36
  145. package/src/contentful/blocks/breadcrumbs/BreadcrumbNavigation.stories.tsx +147 -147
  146. package/src/contentful/blocks/breadcrumbs/index.test.tsx +281 -0
  147. package/src/contentful/blocks/breadcrumbs/index.tsx +95 -95
  148. package/src/contentful/blocks/breadcrumbs/types.ts +8 -8
  149. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  150. package/src/contentful/blocks/button/index.test.tsx +339 -0
  151. package/src/contentful/blocks/button/index.tsx +131 -131
  152. package/src/contentful/blocks/button/types.ts +39 -39
  153. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  154. package/src/contentful/blocks/callout/index.test.tsx +539 -0
  155. package/src/contentful/blocks/callout/index.tsx +277 -277
  156. package/src/contentful/blocks/callout/types.ts +78 -78
  157. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  158. package/src/contentful/blocks/cards/blog-card/index.test.tsx +218 -0
  159. package/src/contentful/blocks/cards/blog-card/index.tsx +129 -129
  160. package/src/contentful/blocks/cards/blog-card/types.ts +34 -34
  161. package/src/contentful/blocks/cards/floating-image-card/index.test.tsx +201 -0
  162. package/src/contentful/blocks/cards/floating-image-card/index.tsx +119 -119
  163. package/src/contentful/blocks/cards/floating-image-card/types.ts +30 -30
  164. package/src/contentful/blocks/cards/full-image-card/index.test.tsx +216 -0
  165. package/src/contentful/blocks/cards/full-image-card/index.tsx +130 -130
  166. package/src/contentful/blocks/cards/full-image-card/types.ts +29 -29
  167. package/src/contentful/blocks/cards/index.test.tsx +39 -0
  168. package/src/contentful/blocks/cards/index.tsx +13 -13
  169. package/src/contentful/blocks/cards/product-card/index.test.tsx +263 -0
  170. package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
  171. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  172. package/src/contentful/blocks/cards/simple-card/index.test.tsx +364 -0
  173. package/src/contentful/blocks/cards/simple-card/index.tsx +325 -325
  174. package/src/contentful/blocks/cards/simple-card/types.ts +71 -71
  175. package/src/contentful/blocks/cards/testimonial-card/index.test.tsx +180 -0
  176. package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
  177. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  178. package/src/contentful/blocks/cards/types.ts +1 -1
  179. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  180. package/src/contentful/blocks/carousel/helper.test.tsx +539 -0
  181. package/src/contentful/blocks/carousel/helper.tsx +494 -494
  182. package/src/contentful/blocks/carousel/index.test.tsx +308 -0
  183. package/src/contentful/blocks/carousel/index.tsx +87 -87
  184. package/src/contentful/blocks/carousel/types.test.ts +16 -0
  185. package/src/contentful/blocks/carousel/types.ts +145 -145
  186. package/src/contentful/blocks/cart-retention-banner/index.test.tsx +409 -0
  187. package/src/contentful/blocks/cart-retention-banner/index.tsx +109 -109
  188. package/src/contentful/blocks/cart-retention-banner/types.ts +11 -11
  189. package/src/contentful/blocks/comparison-table/index.test.tsx +114 -0
  190. package/src/contentful/blocks/comparison-table/index.tsx +29 -29
  191. package/src/contentful/blocks/comparison-table/types.ts +6 -6
  192. package/src/contentful/blocks/cookiebanner/index.test.tsx +277 -0
  193. package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
  194. package/src/contentful/blocks/cookiebanner/type.ts +7 -7
  195. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  196. package/src/contentful/blocks/cta-callout/index.test.tsx +244 -0
  197. package/src/contentful/blocks/cta-callout/index.tsx +73 -73
  198. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  199. package/src/contentful/blocks/dynamic-tabs/index.test.tsx +240 -0
  200. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  201. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  202. package/src/contentful/blocks/email-input-block/index.test.tsx +213 -0
  203. package/src/contentful/blocks/email-input-block/index.tsx +121 -116
  204. package/src/contentful/blocks/email-input-block/types.ts +16 -16
  205. package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
  206. package/src/contentful/blocks/find-kinetic/index.test.tsx +269 -0
  207. package/src/contentful/blocks/find-kinetic/index.tsx +138 -130
  208. package/src/contentful/blocks/find-kinetic/types.ts +20 -19
  209. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  210. package/src/contentful/blocks/floating-banner/index.test.tsx +246 -0
  211. package/src/contentful/blocks/floating-banner/index.tsx +97 -97
  212. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  213. package/src/contentful/blocks/footer/Footer.stories.tsx +317 -317
  214. package/src/contentful/blocks/footer/index.test.tsx +302 -0
  215. package/src/contentful/blocks/footer/index.tsx +91 -91
  216. package/src/contentful/blocks/footer/types.ts +13 -13
  217. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  218. package/src/contentful/blocks/image-promo-bar/helper.test.tsx +61 -0
  219. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  220. package/src/contentful/blocks/image-promo-bar/index.test.tsx +467 -0
  221. package/src/contentful/blocks/image-promo-bar/index.tsx +246 -246
  222. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  223. package/src/contentful/blocks/image-promo-bar/vimeo-embed.test.tsx +142 -0
  224. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  225. package/src/contentful/blocks/image-promo-bar/youtube-embed.test.tsx +104 -0
  226. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  227. package/src/contentful/blocks/modal/constants.ts +53 -53
  228. package/src/contentful/blocks/modal/index.test.tsx +209 -0
  229. package/src/contentful/blocks/modal/index.tsx +108 -108
  230. package/src/contentful/blocks/modal/types.ts +12 -12
  231. package/src/contentful/blocks/navigation/Navigation.stories.mocks.tsx +78 -0
  232. package/src/contentful/blocks/navigation/Navigation.stories.tsx +138 -0
  233. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.test.tsx +208 -0
  234. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +141 -139
  235. package/src/contentful/blocks/navigation/index.test.tsx +924 -0
  236. package/src/contentful/blocks/navigation/index.tsx +569 -568
  237. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.test.tsx +131 -0
  238. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  239. package/src/contentful/blocks/navigation/types.ts +71 -71
  240. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  241. package/src/contentful/blocks/primary-hero/index.test.tsx +286 -0
  242. package/src/contentful/blocks/primary-hero/index.tsx +239 -236
  243. package/src/contentful/blocks/primary-hero/types.ts +37 -37
  244. package/src/contentful/blocks/search-block/index.test.tsx +268 -0
  245. package/src/contentful/blocks/search-block/index.tsx +90 -90
  246. package/src/contentful/blocks/search-block/types.ts +15 -15
  247. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  248. package/src/contentful/blocks/shape-background-wrapper/index.test.tsx +284 -0
  249. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  250. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  251. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  252. package/src/contentful/blocks/text/index.test.tsx +36 -0
  253. package/src/contentful/blocks/text/index.tsx +12 -12
  254. package/src/contentful/blocks/text/types.ts +1 -1
  255. package/src/contentful/index.test.ts +45 -0
  256. package/src/contentful/index.ts +105 -105
  257. package/src/global-mocks/contentful/to-document.ts +25 -0
  258. package/src/global-mocks/cookie.ts +48 -0
  259. package/src/global-mocks/cx.ts +37 -0
  260. package/src/global-mocks/index.ts +89 -0
  261. package/src/global-mocks/speed-card-bg.ts +27 -0
  262. package/src/global-mocks/utm.ts +49 -0
  263. package/src/hooks/contentful/use-contentful-rich-text.test.tsx +1758 -0
  264. package/src/hooks/contentful/use-contentful-rich-text.tsx +309 -309
  265. package/src/hooks/contentful/use-processed-check-list.test.tsx +277 -0
  266. package/src/hooks/contentful/use-processed-check-list.ts +63 -63
  267. package/src/hooks/use-body-scroll-lock.test.ts +134 -0
  268. package/src/hooks/use-body-scroll-lock.ts +34 -34
  269. package/src/hooks/use-carousel-swipe.test.ts +393 -0
  270. package/src/hooks/use-carousel-swipe.ts +264 -264
  271. package/src/hooks/use-outside-click.test.ts +142 -0
  272. package/src/hooks/use-outside-click.ts +17 -17
  273. package/src/index.ts +107 -107
  274. package/src/next/index.test.ts +7 -0
  275. package/src/next/index.ts +5 -5
  276. package/src/setupTests.ts +52 -46
  277. package/src/stories/DocsTemplate.tsx +24 -24
  278. package/src/styles/globals.css +343 -343
  279. package/src/types/global.d.ts +9 -9
  280. package/src/types/micro-components.ts +99 -99
  281. package/src/types/utm.ts +49 -49
  282. package/src/utils/contentful/to-document.test.ts +85 -0
  283. package/src/utils/contentful/to-document.ts +24 -24
  284. package/src/utils/cookie.test.ts +180 -0
  285. package/src/utils/cookie.ts +84 -84
  286. package/src/utils/cx.test.ts +90 -0
  287. package/src/utils/cx.ts +49 -49
  288. package/src/utils/index.test.ts +115 -0
  289. package/src/utils/index.ts +41 -41
  290. package/src/utils/speed-card-bg.test.ts +46 -0
  291. package/src/utils/speed-card-bg.ts +24 -24
  292. package/src/utils/utm.test.ts +359 -0
  293. package/src/utils/utm.ts +221 -221
@@ -0,0 +1,393 @@
1
+ import React from "react";
2
+ import { CarouselSwipeConfig, useCarouselSwipe } from "./use-carousel-swipe";
3
+
4
+ import { act, renderHook } from "@testing-library/react";
5
+
6
+ // Helper to create a mock TouchEvent
7
+ function makeTouchEvent(clientX: number): React.TouchEvent {
8
+ return {
9
+ touches: [{ clientX }],
10
+ } as unknown as React.TouchEvent;
11
+ }
12
+
13
+ describe("useCarouselSwipe", () => {
14
+ const defaultConfig: CarouselSwipeConfig = {
15
+ itemCount: 5,
16
+ enableAutoScroll: false,
17
+ };
18
+
19
+ beforeEach(() => {
20
+ jest.useFakeTimers();
21
+ });
22
+
23
+ afterEach(() => {
24
+ jest.useRealTimers();
25
+ });
26
+
27
+ describe("Initial state", () => {
28
+ it("returns currentIndex 0", () => {
29
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
30
+ expect(result.current.currentIndex).toBe(0);
31
+ });
32
+
33
+ it("returns swipeOffset 0", () => {
34
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
35
+ expect(result.current.swipeOffset).toBe(0);
36
+ });
37
+
38
+ it("returns isSwiping false", () => {
39
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
40
+ expect(result.current.isSwiping).toBe(false);
41
+ });
42
+
43
+ it("returns containerRef", () => {
44
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
45
+ expect(result.current.containerRef).toBeDefined();
46
+ expect(result.current.containerRef.current).toBeNull();
47
+ });
48
+
49
+ it("returns default constants", () => {
50
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
51
+ expect(result.current.constants).toEqual({
52
+ CARD_OFFSET_PERCENTAGE: 105,
53
+ SWIPE_THRESHOLD: 0.15,
54
+ MOBILE_BREAKPOINT: 768,
55
+ AUTO_SCROLL_INTERVAL: 8000,
56
+ });
57
+ });
58
+
59
+ it("uses custom constants from config", () => {
60
+ const { result } = renderHook(() =>
61
+ useCarouselSwipe({
62
+ itemCount: 3,
63
+ cardOffsetPercentage: 110,
64
+ swipeThreshold: 0.2,
65
+ mobileBreakpoint: 1024,
66
+ autoScrollInterval: 5000,
67
+ })
68
+ );
69
+ expect(result.current.constants).toEqual({
70
+ CARD_OFFSET_PERCENTAGE: 110,
71
+ SWIPE_THRESHOLD: 0.2,
72
+ MOBILE_BREAKPOINT: 1024,
73
+ AUTO_SCROLL_INTERVAL: 5000,
74
+ });
75
+ });
76
+ });
77
+
78
+ describe("nextSlide", () => {
79
+ it("advances to next slide", () => {
80
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
81
+ act(() => result.current.nextSlide());
82
+ expect(result.current.currentIndex).toBe(1);
83
+ });
84
+
85
+ it("wraps around to 0 from last slide", () => {
86
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
87
+ // Go to last slide (index 4)
88
+ act(() => result.current.goToSlide(4));
89
+ act(() => result.current.nextSlide());
90
+ expect(result.current.currentIndex).toBe(0);
91
+ });
92
+
93
+ it("does nothing when itemCount is 0", () => {
94
+ const { result } = renderHook(() =>
95
+ useCarouselSwipe({ itemCount: 0, enableAutoScroll: false })
96
+ );
97
+ act(() => result.current.nextSlide());
98
+ expect(result.current.currentIndex).toBe(0);
99
+ });
100
+ });
101
+
102
+ describe("prevSlide", () => {
103
+ it("goes to previous slide", () => {
104
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
105
+ act(() => result.current.goToSlide(2));
106
+ act(() => result.current.prevSlide());
107
+ expect(result.current.currentIndex).toBe(1);
108
+ });
109
+
110
+ it("wraps to last slide from first slide", () => {
111
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
112
+ act(() => result.current.prevSlide());
113
+ expect(result.current.currentIndex).toBe(4);
114
+ });
115
+
116
+ it("does nothing when itemCount is 0", () => {
117
+ const { result } = renderHook(() =>
118
+ useCarouselSwipe({ itemCount: 0, enableAutoScroll: false })
119
+ );
120
+ act(() => result.current.prevSlide());
121
+ expect(result.current.currentIndex).toBe(0);
122
+ });
123
+ });
124
+
125
+ describe("goToSlide", () => {
126
+ it("navigates to specific slide index", () => {
127
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
128
+ act(() => result.current.goToSlide(3));
129
+ expect(result.current.currentIndex).toBe(3);
130
+ });
131
+
132
+ it("ignores negative index", () => {
133
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
134
+ act(() => result.current.goToSlide(-1));
135
+ expect(result.current.currentIndex).toBe(0);
136
+ });
137
+
138
+ it("ignores index >= itemCount", () => {
139
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
140
+ act(() => result.current.goToSlide(5));
141
+ expect(result.current.currentIndex).toBe(0);
142
+ });
143
+
144
+ it("navigates to first slide (index 0)", () => {
145
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
146
+ act(() => result.current.goToSlide(3));
147
+ act(() => result.current.goToSlide(0));
148
+ expect(result.current.currentIndex).toBe(0);
149
+ });
150
+
151
+ it("navigates to last valid slide", () => {
152
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
153
+ act(() => result.current.goToSlide(4));
154
+ expect(result.current.currentIndex).toBe(4);
155
+ });
156
+ });
157
+
158
+ describe("Touch gesture handling", () => {
159
+ it("sets isSwiping to true on touch start", () => {
160
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
161
+ act(() => result.current.handleTouchStart(makeTouchEvent(200)));
162
+ expect(result.current.isSwiping).toBe(true);
163
+ });
164
+
165
+ it("tracks swipe offset during touch move", () => {
166
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
167
+ act(() => result.current.handleTouchStart(makeTouchEvent(200)));
168
+ act(() => result.current.handleTouchMove(makeTouchEvent(150)));
169
+ expect(result.current.swipeOffset).toBe(-50);
170
+ });
171
+
172
+ it("does not track move when not swiping", () => {
173
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
174
+ // Don't call handleTouchStart, so isSwiping is false
175
+ act(() => result.current.handleTouchMove(makeTouchEvent(150)));
176
+ expect(result.current.swipeOffset).toBe(0);
177
+ });
178
+
179
+ it("resets swipeOffset and isSwiping on touch end", () => {
180
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
181
+ act(() => result.current.handleTouchStart(makeTouchEvent(200)));
182
+ act(() => result.current.handleTouchMove(makeTouchEvent(210)));
183
+ act(() => result.current.handleTouchEnd());
184
+ expect(result.current.swipeOffset).toBe(0);
185
+ expect(result.current.isSwiping).toBe(false);
186
+ });
187
+
188
+ it("triggers prevSlide on right swipe exceeding threshold", () => {
189
+ const { result } = renderHook(() =>
190
+ useCarouselSwipe({ ...defaultConfig, swipeThreshold: 0.01 })
191
+ );
192
+ // Start at slide 2 so prevSlide can go back
193
+ act(() => result.current.goToSlide(2));
194
+ act(() => result.current.handleTouchStart(makeTouchEvent(100)));
195
+ act(() => result.current.handleTouchMove(makeTouchEvent(300)));
196
+ act(() => result.current.handleTouchEnd());
197
+ expect(result.current.currentIndex).toBe(1);
198
+ });
199
+
200
+ it("triggers nextSlide on left swipe exceeding threshold", () => {
201
+ const { result } = renderHook(() =>
202
+ useCarouselSwipe({ ...defaultConfig, swipeThreshold: 0.01 })
203
+ );
204
+ act(() => result.current.handleTouchStart(makeTouchEvent(300)));
205
+ act(() => result.current.handleTouchMove(makeTouchEvent(100)));
206
+ act(() => result.current.handleTouchEnd());
207
+ expect(result.current.currentIndex).toBe(1);
208
+ });
209
+
210
+ it("does not change slide when swipe is below threshold", () => {
211
+ const { result } = renderHook(() =>
212
+ useCarouselSwipe({ ...defaultConfig, swipeThreshold: 0.99 })
213
+ );
214
+ act(() => result.current.handleTouchStart(makeTouchEvent(200)));
215
+ act(() => result.current.handleTouchMove(makeTouchEvent(195)));
216
+ act(() => result.current.handleTouchEnd());
217
+ expect(result.current.currentIndex).toBe(0);
218
+ });
219
+ });
220
+
221
+ describe("Auto-scroll", () => {
222
+ it("auto-scrolls to next slide at configured interval", () => {
223
+ const { result } = renderHook(() =>
224
+ useCarouselSwipe({
225
+ itemCount: 5,
226
+ enableAutoScroll: true,
227
+ autoScrollInterval: 3000,
228
+ })
229
+ );
230
+ expect(result.current.currentIndex).toBe(0);
231
+ act(() => jest.advanceTimersByTime(3000));
232
+ expect(result.current.currentIndex).toBe(1);
233
+ act(() => jest.advanceTimersByTime(3000));
234
+ expect(result.current.currentIndex).toBe(2);
235
+ });
236
+
237
+ it("does not auto-scroll when enableAutoScroll is false", () => {
238
+ const { result } = renderHook(() =>
239
+ useCarouselSwipe({
240
+ itemCount: 5,
241
+ enableAutoScroll: false,
242
+ autoScrollInterval: 3000,
243
+ })
244
+ );
245
+ act(() => jest.advanceTimersByTime(10000));
246
+ expect(result.current.currentIndex).toBe(0);
247
+ });
248
+
249
+ it("does not auto-scroll when autoScrollInterval is 0", () => {
250
+ const { result } = renderHook(() =>
251
+ useCarouselSwipe({
252
+ itemCount: 5,
253
+ enableAutoScroll: true,
254
+ autoScrollInterval: 0,
255
+ })
256
+ );
257
+ act(() => jest.advanceTimersByTime(10000));
258
+ expect(result.current.currentIndex).toBe(0);
259
+ });
260
+
261
+ it("does not auto-scroll when itemCount is 0", () => {
262
+ const { result } = renderHook(() =>
263
+ useCarouselSwipe({
264
+ itemCount: 0,
265
+ enableAutoScroll: true,
266
+ autoScrollInterval: 3000,
267
+ })
268
+ );
269
+ act(() => jest.advanceTimersByTime(10000));
270
+ expect(result.current.currentIndex).toBe(0);
271
+ });
272
+
273
+ it("pauses auto-scroll on touch start", () => {
274
+ const { result } = renderHook(() =>
275
+ useCarouselSwipe({
276
+ itemCount: 5,
277
+ enableAutoScroll: true,
278
+ autoScrollInterval: 3000,
279
+ })
280
+ );
281
+ act(() => result.current.handleTouchStart(makeTouchEvent(200)));
282
+ act(() => jest.advanceTimersByTime(10000));
283
+ // Should still be at 0 because auto-scroll was paused
284
+ // (though the initial interval may have already ticked)
285
+ expect(result.current.isSwiping).toBe(true);
286
+ });
287
+
288
+ it("restarts auto-scroll after touch end when enabled", () => {
289
+ const { result } = renderHook(() =>
290
+ useCarouselSwipe({
291
+ itemCount: 5,
292
+ enableAutoScroll: true,
293
+ autoScrollInterval: 3000,
294
+ })
295
+ );
296
+ // Touch interaction
297
+ act(() => result.current.handleTouchStart(makeTouchEvent(200)));
298
+ act(() => result.current.handleTouchEnd());
299
+ // After touch end, auto-scroll should restart
300
+ act(() => jest.advanceTimersByTime(3000));
301
+ expect(result.current.currentIndex).toBeGreaterThan(0);
302
+ });
303
+
304
+ it("cleans up interval on unmount", () => {
305
+ const clearIntervalSpy = jest.spyOn(global, "clearInterval");
306
+ const { unmount } = renderHook(() =>
307
+ useCarouselSwipe({
308
+ itemCount: 5,
309
+ enableAutoScroll: true,
310
+ autoScrollInterval: 3000,
311
+ })
312
+ );
313
+ unmount();
314
+ expect(clearIntervalSpy).toHaveBeenCalled();
315
+ clearIntervalSpy.mockRestore();
316
+ });
317
+ });
318
+
319
+ describe("Responsive behavior", () => {
320
+ it("detects mobile viewport", () => {
321
+ Object.defineProperty(window, "innerWidth", {
322
+ writable: true,
323
+ value: 500,
324
+ });
325
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
326
+ expect(result.current.isMobile).toBe(true);
327
+ });
328
+
329
+ it("detects desktop viewport", () => {
330
+ Object.defineProperty(window, "innerWidth", {
331
+ writable: true,
332
+ value: 1024,
333
+ });
334
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
335
+ expect(result.current.isMobile).toBe(false);
336
+ });
337
+
338
+ it("updates isMobile on resize", () => {
339
+ Object.defineProperty(window, "innerWidth", {
340
+ writable: true,
341
+ value: 1024,
342
+ });
343
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
344
+ expect(result.current.isMobile).toBe(false);
345
+
346
+ act(() => {
347
+ Object.defineProperty(window, "innerWidth", {
348
+ writable: true,
349
+ value: 500,
350
+ });
351
+ window.dispatchEvent(new Event("resize"));
352
+ });
353
+ expect(result.current.isMobile).toBe(true);
354
+ });
355
+
356
+ it("updates containerWidth on resize", () => {
357
+ const { result } = renderHook(() => useCarouselSwipe(defaultConfig));
358
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
359
+ const initialWidth = result.current.containerWidth;
360
+ act(() => {
361
+ Object.defineProperty(window, "innerWidth", {
362
+ writable: true,
363
+ value: 800,
364
+ });
365
+ window.dispatchEvent(new Event("resize"));
366
+ });
367
+ // containerRef is null so falls back to window.innerWidth
368
+ expect(result.current.containerWidth).toBe(800);
369
+ });
370
+
371
+ it("uses custom mobileBreakpoint", () => {
372
+ Object.defineProperty(window, "innerWidth", {
373
+ writable: true,
374
+ value: 900,
375
+ });
376
+ const { result } = renderHook(() =>
377
+ useCarouselSwipe({ ...defaultConfig, mobileBreakpoint: 1024 })
378
+ );
379
+ expect(result.current.isMobile).toBe(true);
380
+ });
381
+
382
+ it("removes resize listeners on unmount", () => {
383
+ const removeEventListenerSpy = jest.spyOn(window, "removeEventListener");
384
+ const { unmount } = renderHook(() => useCarouselSwipe(defaultConfig));
385
+ unmount();
386
+ const resizeCalls = removeEventListenerSpy.mock.calls.filter(
387
+ call => call[0] === "resize"
388
+ );
389
+ expect(resizeCalls.length).toBeGreaterThanOrEqual(2); // containerWidth + isMobile
390
+ removeEventListenerSpy.mockRestore();
391
+ });
392
+ });
393
+ });