@cnamts/synapse 1.0.25 → 1.0.27

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 (393) hide show
  1. package/dist/{AutocompleteFilter-D7qBuCAP.js → AutocompleteFilter-C9eLKyW8.js} +3 -3
  2. package/dist/{DateFilter-BitMWrMU.js → DateFilter-y-GLkAkn.js} +9 -9
  3. package/dist/{NumberFilter-BTLUxw0a.js → NumberFilter-DN6hIBS7.js} +1 -1
  4. package/dist/{PeriodFilter-B5rUIPAC.js → PeriodFilter-MoUUp9qS.js} +1 -1
  5. package/dist/{SelectFilter-l4QnRcuk.js → SelectFilter-bCbrdLmu.js} +1 -1
  6. package/dist/{TextFilter-C9hj6Qrp.js → TextFilter-CvjgEaoM.js} +4 -4
  7. package/dist/apLightTheme2026-ug4Y23ns.js +611 -0
  8. package/dist/components/Customs/Selects/SyAutocomplete/SyAutocomplete.d.ts +2369 -351
  9. package/dist/components/Customs/Selects/SyAutocomplete/composables/useSyAutocompleteValidation.d.ts +18 -0
  10. package/dist/components/Customs/Selects/SyAutocomplete/utils/ariaManager.d.ts +1 -1
  11. package/dist/components/Customs/Selects/SyAutocomplete/utils/useKeyboardHandler.d.ts +3 -1
  12. package/dist/components/Customs/Selects/SySelect/SySelect.d.ts +58 -288
  13. package/dist/components/Customs/Selects/SySelect/composables/useSySelectKeyboard.d.ts +1 -0
  14. package/dist/components/Customs/Selects/SySelect/composables/useSySelectValidation.d.ts +15 -0
  15. package/dist/components/Customs/SyCheckBoxGroup/SyCheckBoxGroup.d.ts +4 -3
  16. package/dist/components/Customs/SyCheckbox/SyCheckbox.d.ts +4 -3
  17. package/dist/components/Customs/SyIconButton/SyIconButton.d.ts +18 -0
  18. package/dist/components/Customs/SyRadioGroup/SyRadioGroup.d.ts +20 -37
  19. package/dist/components/Customs/SyRadioGroup/composables/useSyRadioGroupValidation.d.ts +50 -0
  20. package/dist/components/Customs/SyTextField/SyTextField.d.ts +8 -10
  21. package/dist/components/DatePicker/CalendarMode/DatePicker.d.ts +197 -185
  22. package/dist/components/DatePicker/ComplexDatePicker/ComplexDatePicker.d.ts +91 -82
  23. package/dist/components/DatePicker/DateTextInput/DateTextInput.d.ts +35 -32
  24. package/dist/components/DatePicker/composables/index.d.ts +1 -0
  25. package/dist/components/DatePicker/composables/useDatePickerState.d.ts +3 -3
  26. package/dist/components/DatePicker/composables/useDatePickerValidationBridge.d.ts +51 -0
  27. package/dist/components/DatePicker/composables/useDateTextField.d.ts +2 -2
  28. package/dist/components/DatePicker/composables/useInputBlurHandler.d.ts +2 -2
  29. package/dist/components/DatePicker/types.d.ts +1 -2
  30. package/dist/components/LunarCalendar/useLunarCalendarValidation.d.ts +1 -0
  31. package/dist/components/MonthPicker/MonthPicker.d.ts +24 -24
  32. package/dist/components/MonthPicker/MonthPickerText/MonthPickerInput.d.ts +24 -24
  33. package/dist/components/NirField/NirField.d.ts +64 -60
  34. package/dist/components/NirField/useNirValidation.d.ts +6 -2
  35. package/dist/components/PasswordField/PasswordField.d.ts +3 -3
  36. package/dist/components/PeriodField/PeriodField.d.ts +338 -314
  37. package/dist/components/PhoneField/PhoneField.d.ts +34 -24
  38. package/dist/components/RangeField/RangeSlider/RangeSlider.d.ts +0 -3
  39. package/dist/components/RatingPicker/EmotionPicker/EmotionPicker.d.ts +3 -1
  40. package/dist/components/RatingPicker/NumberPicker/NumberPicker.d.ts +4 -3
  41. package/dist/components/RatingPicker/RatingPicker.d.ts +18 -5
  42. package/dist/components/RatingPicker/StarsPicker/StarsPicker.d.ts +3 -1
  43. package/dist/components/RatingPicker/useRatingFocus.d.ts +18 -0
  44. package/dist/components/SyTextArea/SyTextArea.d.ts +25 -15
  45. package/dist/components/SyTextArea/composables/useSyTextAreaValidation.d.ts +20 -0
  46. package/dist/components/SyTextArea/locales.d.ts +1 -0
  47. package/dist/components/Tables/SyServerTable/SyServerTable.d.ts +5 -4
  48. package/dist/components/Tables/SyTable/SyTable.d.ts +5 -4
  49. package/dist/components/Tables/common/SyTablePagination.d.ts +152 -364
  50. package/dist/components/Tables/common/TableHeader.d.ts +1 -1
  51. package/dist/components/Tables/common/filters/DateFilter.d.ts +4 -4
  52. package/dist/components/Tables/common/types.d.ts +2 -0
  53. package/dist/components/index.d.ts +1 -0
  54. package/dist/composables/date/useDateInitializationDayjs.d.ts +3 -1
  55. package/dist/composables/unifyValidation/documentationValidationProps.d.ts +160 -160
  56. package/dist/composables/unifyValidation/useCustomValidation.d.ts +3 -1
  57. package/dist/composables/unifyValidation/useValidation.d.ts +27 -19
  58. package/dist/composables/unifyValidation/useVuetifyValidation.d.ts +1 -1
  59. package/dist/composables/validation/useValidation.d.ts +1 -0
  60. package/dist/design-system-v3.js +81 -80
  61. package/dist/designTokens/tokens/amelipro/apContextual.d.ts +6 -6
  62. package/dist/designTokens/tokens/amelipro/apDarkTheme.d.ts +3 -1
  63. package/dist/designTokens/tokens/amelipro/apLightTheme.d.ts +53 -98
  64. package/dist/designTokens/tokens/baseContextualTokens.d.ts +0 -6
  65. package/dist/designTokens/tokens/baseTokens.d.ts +232 -0
  66. package/dist/designTokens/tokens/cnam/cnamContextual.d.ts +6 -6
  67. package/dist/designTokens/tokens/cnam/cnamDarkTheme.d.ts +1 -1
  68. package/dist/designTokens/tokens/cnam/cnamLightTheme.d.ts +57 -99
  69. package/dist/designTokens/tokens/pa/paContextual.d.ts +0 -6
  70. package/dist/designTokens/tokens/pa/paDarkTheme.d.ts +1 -1
  71. package/dist/designTokens/tokens/pa/paLightTheme.d.ts +53 -97
  72. package/dist/designTokens/tokens/pa/paSemantic.d.ts +1 -0
  73. package/dist/designTokens/tokens/semanticTokens.d.ts +112 -0
  74. package/dist/{main-Cpx8Co6H.js → main-CI6Q9nmO.js} +13843 -13478
  75. package/dist/synapse.css +1 -1
  76. package/dist/vuetifyConfig.js +208 -72
  77. package/package.json +10 -7
  78. package/src/assets/overrides/_icons.scss +5 -17
  79. package/src/assets/overrides/_otp.scss +4 -5
  80. package/src/assets/overrides/_typography.scss +2 -1
  81. package/src/assets/overrides/_utilities.scss +1 -42
  82. package/src/components/Accordion/Accordion.vue +2 -0
  83. package/src/components/ChipList/ChipList.vue +30 -18
  84. package/src/components/ChipList/tests/chipList.spec.ts +4 -4
  85. package/src/components/CookiesSelection/CookiesInformation/CookiesInformation.vue +2 -1
  86. package/src/components/CookiesSelection/CookiesSelection.vue +2 -1
  87. package/src/components/CopyBtn/CopyBtn.vue +9 -0
  88. package/src/components/Customs/Selects/SelectBtnField/SelectBtnField.stories.ts +4 -0
  89. package/src/components/Customs/Selects/SelectBtnField/SelectBtnField.vue +7 -6
  90. package/src/components/Customs/Selects/SelectBtnField/tests/SelectBtnField.spec.ts +223 -0
  91. package/src/components/Customs/Selects/SyAutocomplete/SyAutocomplete.stories.ts +283 -351
  92. package/src/components/Customs/Selects/SyAutocomplete/SyAutocomplete.vue +183 -219
  93. package/src/components/Customs/Selects/SyAutocomplete/composables/useSyAutocompleteValidation.ts +101 -0
  94. package/src/components/Customs/Selects/SyAutocomplete/tests/SyAutocomplete.spec.ts +761 -1
  95. package/src/components/Customs/Selects/SyAutocomplete/utils/ariaManager.ts +3 -1
  96. package/src/components/Customs/Selects/SyAutocomplete/utils/useKeyboardHandler.ts +79 -5
  97. package/src/components/Customs/Selects/SyAutocomplete/validation/Validation.stories.ts +1029 -0
  98. package/src/components/Customs/Selects/SySelect/SySelect.stories.ts +52 -217
  99. package/src/components/Customs/Selects/SySelect/SySelect.vue +248 -236
  100. package/src/components/Customs/Selects/SySelect/composables/useSySelectKeyboard.ts +3 -0
  101. package/src/components/Customs/Selects/SySelect/composables/useSySelectValidation.ts +64 -0
  102. package/src/components/Customs/Selects/SySelect/tests/SySelect.spec.ts +440 -5
  103. package/src/components/Customs/Selects/SySelect/validation/Validation.stories.ts +1026 -0
  104. package/src/components/Customs/SyCheckBoxGroup/SyCheckBoxGroup.stories.ts +18 -7
  105. package/src/components/Customs/SyCheckBoxGroup/SyCheckBoxGroup.vue +5 -5
  106. package/src/components/Customs/SyCheckbox/SyCheckbox.stories.ts +8 -8
  107. package/src/components/Customs/SyCheckbox/SyCheckbox.vue +27 -6
  108. package/src/components/Customs/SyCheckbox/tests/SyCheckbox.spec.ts +1 -1
  109. package/src/components/Customs/SyIcon/accessibilite/Accessibility.mdx +0 -6
  110. package/src/components/Customs/SyIcon/utils/tests/iconUtils.spec.ts +107 -0
  111. package/src/components/Customs/SyRadioGroup/SyRadioGroup.mdx +2 -2
  112. package/src/components/Customs/SyRadioGroup/SyRadioGroup.stories.ts +395 -200
  113. package/src/components/Customs/SyRadioGroup/SyRadioGroup.vue +100 -127
  114. package/src/components/Customs/SyRadioGroup/composables/useSyRadioGroupValidation.ts +127 -0
  115. package/src/components/Customs/SyRadioGroup/tests/SyRadioGroup.a11y.spec.ts +93 -1
  116. package/src/components/Customs/SyRadioGroup/tests/SyRadioGroup.spec.ts +146 -9
  117. package/src/components/Customs/SyRadioGroup/tests/SyRadioGroup.visual.cy.ts +165 -0
  118. package/src/components/Customs/SyRadioGroup/validation/Validation.stories.ts +773 -0
  119. package/src/components/Customs/SyTabs/SyTabs.stories.ts +5 -5
  120. package/src/components/Customs/SyTabs/config.ts +3 -3
  121. package/src/components/Customs/SyTabs/tests/SyTabs.spec.ts +265 -0
  122. package/src/components/Customs/SyTabs/tests/useTabTransition.spec.ts +188 -0
  123. package/src/components/Customs/SyTextField/SyTextField.stories.ts +10 -29
  124. package/src/components/Customs/SyTextField/SyTextField.vue +52 -17
  125. package/src/components/DataList/DataList.stories.ts +1 -1
  126. package/src/components/DataListItem/tests/DataListItem.spec.ts +3 -1
  127. package/src/components/DatePicker/CalendarMode/DatePicker.stories.ts +1 -1
  128. package/src/components/DatePicker/CalendarMode/DatePicker.vue +52 -154
  129. package/src/components/DatePicker/CalendarMode/tests/DatePicker.coverage.spec.ts +156 -0
  130. package/src/components/DatePicker/CalendarMode/tests/DatePicker.spec.ts +495 -4
  131. package/src/components/DatePicker/ComplexDatePicker/ComplexDatePicker.stories.ts +1 -1
  132. package/src/components/DatePicker/ComplexDatePicker/ComplexDatePicker.vue +55 -73
  133. package/src/components/DatePicker/ComplexDatePicker/tests/ComplexDatePicker.spec.ts +207 -1
  134. package/src/components/DatePicker/ComplexDatePicker/tests/bridge-integration.regression.spec.ts +210 -0
  135. package/src/components/DatePicker/ComplexDatePicker/tests/calendar-navigation.regression.spec.ts +214 -0
  136. package/src/components/DatePicker/ComplexDatePicker/tests/validation-cross.regression.spec.ts +194 -0
  137. package/src/components/DatePicker/ComplexDatePicker/tests/validation-success-messages.regression.spec.ts +83 -0
  138. package/src/components/DatePicker/DatePickerValidationExample/DatePickerValidation.stories.ts +1 -1
  139. package/src/components/DatePicker/DateTextInput/DateTextInput.vue +169 -60
  140. package/src/components/DatePicker/DateTextInput/NoCalendar.stories.ts +1 -1
  141. package/src/components/DatePicker/DateTextInput/tests/DateTextInput.spec.ts +320 -0
  142. package/src/components/DatePicker/composables/index.ts +1 -0
  143. package/src/components/DatePicker/composables/tests/useCalendarKeyboardNavigation.spec.ts +360 -0
  144. package/src/components/DatePicker/composables/tests/useDatePickerValidationBridge.spec.ts +129 -0
  145. package/src/components/DatePicker/composables/useDatePickerState.ts +33 -14
  146. package/src/components/DatePicker/composables/useDatePickerValidationBridge.ts +205 -0
  147. package/src/components/DatePicker/composables/useDateRangeInput.ts +2 -1
  148. package/src/components/DatePicker/composables/useDateSelection.ts +2 -1
  149. package/src/components/DatePicker/composables/useDateTextField.ts +2 -2
  150. package/src/components/DatePicker/composables/useInputBlurHandler.ts +2 -2
  151. package/src/components/DatePicker/docExamples/BidirectionalComplexValidation.vue +1 -1
  152. package/src/components/DatePicker/docExamples/DatePickerBidirectionalValidation.vue +1 -1
  153. package/src/components/DatePicker/tests/exposed-methods.coverage.spec.ts +75 -0
  154. package/src/components/DatePicker/types.ts +1 -2
  155. package/src/components/DialogBox/DialogBox.stories.ts +8 -8
  156. package/src/components/DialogBox/DialogBox.vue +1 -1
  157. package/src/components/DialogBox/accessibilite/Accessibility.mdx +86 -22
  158. package/src/components/FileList/UploadItem/UploadItem.vue +4 -4
  159. package/src/components/FileUpload/FileUpload.vue +2 -2
  160. package/src/components/FileUpload/FileUploadContent.vue +1 -1
  161. package/src/components/FilterInline/FilterInline.mdx +2 -2
  162. package/src/components/FilterSideBar/FilterSideBar.stories.ts +1 -1
  163. package/src/components/FilterSideBar/FilterSideBar.vue +4 -3
  164. package/src/components/FooterBar/FooterBar.vue +7 -7
  165. package/src/components/FranceConnectBtn/FranceConnectBtn.vue +1 -1
  166. package/src/components/HeaderBar/HeaderBurgerMenu/HeaderMenuItem/HeaderMenuItem.vue +2 -2
  167. package/src/components/HeaderBar/HeaderBurgerMenu/HeaderSubMenu/HeaderSubMenu.vue +7 -7
  168. package/src/components/HeaderBar/HeaderMenuBtn/HeaderMenuBtn.vue +2 -2
  169. package/src/components/HeaderLoading/tests/HeaderLoading.spec.ts +87 -8
  170. package/src/components/HeaderNavigationBar/HorizontalNavbar/HorizontalNavbar.vue +3 -3
  171. package/src/components/HeaderNavigationBar/HorizontalNavbar/tests/HorizontalNavbar.spec.ts +589 -0
  172. package/src/components/HeaderToolbar/tests/HeaderToolBar.spec.ts +153 -1
  173. package/src/components/HeaderToolbar/tests/useMobileRightMenu.spec.ts +258 -0
  174. package/src/components/LangBtn/LangBtn.vue +2 -1
  175. package/src/components/LogoBrandSection/tests/LogoBrandSection.spec.ts +2 -2
  176. package/src/components/LogoBrandSection/tests/__snapshots__/LogoBrandSection.spec.ts.snap +1 -1
  177. package/src/components/LunarCalendar/tests/useLunarCalendarRules.spec.ts +184 -0
  178. package/src/components/MonthPicker/MonthPickerVisual/MonthSelector.vue +3 -3
  179. package/src/components/MonthPicker/MonthPickerVisual/VisualPickerFooter.vue +1 -1
  180. package/src/components/MonthPicker/MonthPickerVisual/VisualPickerHeader.vue +2 -2
  181. package/src/components/MonthPicker/MonthPickerVisual/YearSelector.vue +1 -1
  182. package/src/components/NirField/NirField.vue +3 -3
  183. package/src/components/NotificationBar/Notification/Notification.vue +12 -12
  184. package/src/components/NotificationBar/NotificationBar.stories.ts +8 -8
  185. package/src/components/PaginatedTable/PaginatedTable.vue +1 -1
  186. package/src/components/PaginatedTable/Pagination.vue +3 -3
  187. package/src/components/PasswordField/PasswordField.vue +15 -11
  188. package/src/components/PasswordField/tests/PasswordField.spec.ts +3 -3
  189. package/src/components/PhoneField/PhoneField.vue +4 -2
  190. package/src/components/RangeField/RangeSlider/RangeSlider.vue +11 -18
  191. package/src/components/RangeField/RangeSlider/Tooltip/Tooltip.vue +1 -1
  192. package/src/components/RatingPicker/EmotionPicker/EmotionPicker.vue +32 -48
  193. package/src/components/RatingPicker/EmotionPicker/tests/__snapshots__/EmotionPicker.spec.ts.snap +5 -0
  194. package/src/components/RatingPicker/NumberPicker/NumberPicker.vue +48 -53
  195. package/src/components/RatingPicker/NumberPicker/tests/NumberPicker.spec.ts +2 -1
  196. package/src/components/RatingPicker/NumberPicker/tests/__snapshots__/NumberPicker.spec.ts.snap +40 -13
  197. package/src/components/RatingPicker/RatingPicker.stories.ts +65 -88
  198. package/src/components/RatingPicker/RatingPicker.vue +71 -15
  199. package/src/components/RatingPicker/StarsPicker/StarsPicker.vue +28 -37
  200. package/src/components/RatingPicker/StarsPicker/tests/StarsPicker.spec.ts +1 -1
  201. package/src/components/RatingPicker/StarsPicker/tests/__snapshots__/StarsPicker.spec.ts.snap +5 -0
  202. package/src/components/RatingPicker/accessibilite/Accessibility.mdx +137 -9
  203. package/src/components/RatingPicker/tests/RatingPicker.a11y.spec.ts +123 -0
  204. package/src/components/RatingPicker/tests/RatingPicker.spec.ts +3 -2
  205. package/src/components/RatingPicker/tests/__snapshots__/RatingPicker.spec.ts.snap +40 -11
  206. package/src/components/RatingPicker/useRatingFocus.ts +97 -0
  207. package/src/components/StatusPage/tests/StatusPage.spec.ts +149 -0
  208. package/src/components/SubHeader/SubHeader.vue +1 -1
  209. package/src/components/SyAlert/SyAlert.vue +23 -23
  210. package/src/components/SyTextArea/SyTextArea.stories.ts +177 -131
  211. package/src/components/SyTextArea/SyTextArea.vue +257 -74
  212. package/src/components/SyTextArea/composables/useSyTextAreaValidation.ts +81 -0
  213. package/src/components/SyTextArea/locales.ts +1 -0
  214. package/src/components/SyTextArea/tests/SyTextArea.spec.ts +449 -1
  215. package/src/components/SyTextArea/useDefaultValidationRules.ts +2 -7
  216. package/src/components/SyTextArea/validation/Validation.stories.ts +856 -0
  217. package/src/components/TableToolbar/TableToolbar.vue +6 -6
  218. package/src/components/TableToolbar/accessibilite/Accessibility.mdx +81 -7
  219. package/src/components/Tables/SyServerTable/SyServerTable.stories.ts +163 -0
  220. package/src/components/Tables/SyServerTable/SyServerTable.vue +3 -2
  221. package/src/components/Tables/SyServerTable/tests/SyServerTable.spec.ts +67 -0
  222. package/src/components/Tables/SyTable/SyTable.stories.ts +94 -0
  223. package/src/components/Tables/SyTable/SyTable.vue +3 -2
  224. package/src/components/Tables/SyTable/tests/SyTable.spec.ts +64 -0
  225. package/src/components/Tables/common/SyTableFilter.vue +4 -4
  226. package/src/components/Tables/common/SyTablePagination.vue +1 -0
  227. package/src/components/Tables/common/TableHeader.vue +3 -3
  228. package/src/components/Tables/common/filters/DateFilter.vue +2 -2
  229. package/src/components/Tables/common/filters/logics/tests/NumberFilterLogic.spec.ts +176 -0
  230. package/src/components/Tables/common/filters/logics/tests/SelectFilterLogic.spec.ts +111 -0
  231. package/src/components/Tables/common/tableStyles.scss +6 -6
  232. package/src/components/Tables/common/types.ts +2 -0
  233. package/src/components/UploadWorkflow/tests/UploadWorkflow.spec.ts +2 -0
  234. package/src/components/index.ts +1 -0
  235. package/src/composables/date/tests/useDateFormatDayjs.spec.ts +112 -0
  236. package/src/composables/date/tests/{useDateInitialization.spec.ts → useDateInitializationDayjs.spec.ts} +39 -3
  237. package/src/composables/date/tests/useHolidayDay.spec.ts +109 -0
  238. package/src/composables/date/useDateInitializationDayjs.ts +4 -1
  239. package/src/composables/rules/tests/useFieldValidation.spec.ts +374 -0
  240. package/src/composables/tests/useError.spec.ts +30 -0
  241. package/src/composables/tests/useFormFieldErrorHandling.spec.ts +234 -0
  242. package/src/composables/unifyValidation/documentationValidationProps.ts +12 -12
  243. package/src/composables/unifyValidation/tests/documentationValidationProps.spec.ts +177 -0
  244. package/src/composables/unifyValidation/tests/useCustomValidation.spec.ts +32 -1
  245. package/src/composables/unifyValidation/tests/useValidation.spec.ts +28 -2
  246. package/src/composables/unifyValidation/useCustomValidation.ts +34 -12
  247. package/src/composables/unifyValidation/useValidation.ts +55 -27
  248. package/src/composables/unifyValidation/useVuetifyValidation.ts +2 -2
  249. package/src/composables/useFilterable/useFilterable.spec.ts +42 -0
  250. package/src/composables/useFilterable/useFilterable.ts +11 -7
  251. package/src/composables/useFormFieldErrorHandling.ts +6 -3
  252. package/src/composables/validation/tests/useValidation.spec.ts +2 -2
  253. package/src/composables/validation/useValidation.ts +15 -3
  254. package/src/composantsVuetify/VBtn/VBtn.mdx +9 -39
  255. package/src/composantsVuetify/VBtn/v-btn.stories.ts +26 -86
  256. package/src/composantsVuetify/VCard/VCard.mdx +59 -0
  257. package/src/composantsVuetify/VCard/v-card.stories.ts +279 -0
  258. package/src/designTokens/tokens/amelipro/apColors2026.ts +1 -1
  259. package/src/designTokens/tokens/amelipro/apContextual.ts +6 -0
  260. package/src/designTokens/tokens/amelipro/apDarkTheme.ts +2 -2
  261. package/src/designTokens/tokens/amelipro/apLightTheme.ts +72 -100
  262. package/src/designTokens/tokens/amelipro/apSemantic.ts +1 -1
  263. package/src/designTokens/tokens/baseContextualTokens.ts +1 -6
  264. package/src/designTokens/tokens/baseTokens.ts +232 -0
  265. package/src/designTokens/tokens/cnam/cnamContextual.ts +6 -0
  266. package/src/designTokens/tokens/cnam/cnamDarkTheme.ts +2 -2
  267. package/src/designTokens/tokens/cnam/cnamLightTheme.ts +76 -101
  268. package/src/designTokens/tokens/pa/paDarkTheme.ts +2 -2
  269. package/src/designTokens/tokens/pa/paLightTheme.ts +73 -99
  270. package/src/designTokens/tokens/pa/paSemantic.ts +2 -0
  271. package/src/designTokens/tokens/semanticTokens.ts +114 -0
  272. package/src/stories/Accessibilite/Aculturation/SensibilisationAccessibilite.mdx +61 -91
  273. package/src/stories/Accessibilite/AuditDesignSystem.mdx +5 -8
  274. package/src/stories/Accessibilite/AuditEtContreAudit/Exemptions-derogations.mdx +1 -1
  275. package/src/stories/Accessibilite/AuditEtContreAudit/Introduction.mdx +11 -8
  276. package/src/stories/Accessibilite/AuditEtContreAudit/RGAA.mdx +6 -7
  277. package/src/stories/Accessibilite/Introduction.mdx +30 -30
  278. package/src/stories/Accessibilite/KitDePreAudit/Echantillonnage.mdx +168 -78
  279. package/src/stories/Accessibilite/KitDePreAudit/Introduction.mdx +13 -6
  280. package/src/stories/Accessibilite/KitDePreAudit/Outils/Introduction.mdx +66 -45
  281. package/src/stories/Accessibilite/KitDePreAudit/Outils/LecteursDEcran.mdx +23 -49
  282. package/src/stories/Accessibilite/KitDePreAudit/Outils/Tanaguru/FauxPositifs.stories.ts +6 -0
  283. package/src/stories/Accessibilite/KitDePreAudit/Outils/Tanaguru/Utilisation.mdx +7 -19
  284. package/src/stories/Accessibilite/KitDePreAudit/Preaudit.mdx +18 -20
  285. package/src/stories/Components/Components.stories.ts +59 -6
  286. package/src/stories/DesignTokens/ColorIntegrationExample.vue +2 -3
  287. package/src/stories/DesignTokens/Colors.mdx +6 -8
  288. package/src/stories/DesignTokens/colors.stories.ts +244 -1081
  289. package/src/utils/amelipro/toKebabCase/tests/toKebabCase.spec.ts +52 -0
  290. package/src/utils/formatNir/tests/formatNir.spec.ts +34 -0
  291. package/src/utils/tests/insertAt.spec.ts +44 -0
  292. package/dist/AutocompleteFilter-Df9i5mAl.cjs +0 -1
  293. package/dist/DateFilter-BJD6FMev.cjs +0 -1
  294. package/dist/NumberFilter-DGCzCXzI.cjs +0 -1
  295. package/dist/PeriodFilter-DO_ecTZW.cjs +0 -1
  296. package/dist/SelectFilter-CGwcKWLm.cjs +0 -1
  297. package/dist/TextFilter-B8nf7xoK.cjs +0 -1
  298. package/dist/apLightTheme-CEK4iY3f.cjs +0 -1
  299. package/dist/apLightTheme-DnIM24Lv.js +0 -950
  300. package/dist/composables/date/useDateFormat.d.ts +0 -26
  301. package/dist/composables/date/useDateInitialization.d.ts +0 -18
  302. package/dist/design-system-v3.umd.cjs +0 -1
  303. package/dist/main-ByDPHpae.cjs +0 -1067
  304. package/dist/tooth-11-D3sLWv2n.cjs +0 -1
  305. package/dist/tooth-12-CXrLuH03.cjs +0 -1
  306. package/dist/tooth-13-BSfo5fpT.cjs +0 -1
  307. package/dist/tooth-14-DMzulx0h.cjs +0 -1
  308. package/dist/tooth-15-BKRFVi-9.cjs +0 -1
  309. package/dist/tooth-16-CpuxAbuM.cjs +0 -1
  310. package/dist/tooth-17-BPoahUdg.cjs +0 -1
  311. package/dist/tooth-18-DhHJz8sy.cjs +0 -1
  312. package/dist/tooth-21-Dgd5hn_X.cjs +0 -1
  313. package/dist/tooth-22-C2Tn19sB.cjs +0 -1
  314. package/dist/tooth-23-C9uaaSGb.cjs +0 -1
  315. package/dist/tooth-24-BrK9UGpf.cjs +0 -1
  316. package/dist/tooth-25-CE_EfGNp.cjs +0 -1
  317. package/dist/tooth-26-Ctv4i9Fy.cjs +0 -1
  318. package/dist/tooth-27-C5J7JkWM.cjs +0 -1
  319. package/dist/tooth-28-Z9oWqjo0.cjs +0 -1
  320. package/dist/tooth-31-BrYqmkTi.cjs +0 -1
  321. package/dist/tooth-32-BNNR0oCZ.cjs +0 -1
  322. package/dist/tooth-33-DuxvqO2J.cjs +0 -1
  323. package/dist/tooth-34-BCSCXMB6.cjs +0 -1
  324. package/dist/tooth-35-BLUXkX88.cjs +0 -1
  325. package/dist/tooth-36-IrKHYqlA.cjs +0 -1
  326. package/dist/tooth-37-BYqpdMwo.cjs +0 -1
  327. package/dist/tooth-38-B_eNXXdu.cjs +0 -1
  328. package/dist/tooth-41-Ddva4Ot8.cjs +0 -1
  329. package/dist/tooth-42-szcDqlM0.cjs +0 -1
  330. package/dist/tooth-43-B3ka6rQm.cjs +0 -1
  331. package/dist/tooth-44-CazyQucj.cjs +0 -1
  332. package/dist/tooth-45-B4HQtc8n.cjs +0 -1
  333. package/dist/tooth-46-BPM40gbG.cjs +0 -1
  334. package/dist/tooth-47-Dvr20dlh.cjs +0 -1
  335. package/dist/tooth-48-Bd8ljGsF.cjs +0 -1
  336. package/dist/tooth-51-OBpwCOF3.cjs +0 -1
  337. package/dist/tooth-52-aKxyHcmq.cjs +0 -1
  338. package/dist/tooth-53-vCwJjTOc.cjs +0 -1
  339. package/dist/tooth-54-DsWu2iFy.cjs +0 -1
  340. package/dist/tooth-55-BxC1X2Dn.cjs +0 -1
  341. package/dist/tooth-61-BbLvxMQi.cjs +0 -1
  342. package/dist/tooth-62-CmTkWczP.cjs +0 -1
  343. package/dist/tooth-63-DI7l_2qI.cjs +0 -1
  344. package/dist/tooth-64-B21sOsJh.cjs +0 -1
  345. package/dist/tooth-65-D2ZC2VEr.cjs +0 -1
  346. package/dist/tooth-71-D473PPO5.cjs +0 -1
  347. package/dist/tooth-72-Drh1wnNu.cjs +0 -1
  348. package/dist/tooth-73-DzlwYI23.cjs +0 -1
  349. package/dist/tooth-74-8aGvcZPg.cjs +0 -1
  350. package/dist/tooth-75-BFK7At_r.cjs +0 -1
  351. package/dist/tooth-81-BZmR-I0M.cjs +0 -1
  352. package/dist/tooth-82-euVfUUZV.cjs +0 -1
  353. package/dist/tooth-83-KV010j64.cjs +0 -1
  354. package/dist/tooth-84-BBg1RjhZ.cjs +0 -1
  355. package/dist/tooth-85-Cr-kc1wM.cjs +0 -1
  356. package/dist/vuetifyConfig.umd.cjs +0 -1
  357. package/src/components/BackBtn/tests/__snapshots__/back-btn-custom-bg.snap.png +0 -0
  358. package/src/components/BackBtn/tests/__snapshots__/back-btn-dark-mode.snap.png +0 -0
  359. package/src/components/BackBtn/tests/__snapshots__/back-btn-default.snap.png +0 -0
  360. package/src/components/BackBtn/tests/__snapshots__/back-btn-no-icon.snap.png +0 -0
  361. package/src/components/DatePicker/CalendarMode/tests/DatePicker.events.spec.ts +0 -178
  362. package/src/components/DialogBox/tests/__snapshots__/dialog-box-custom-texts.snap.png +0 -0
  363. package/src/components/DialogBox/tests/__snapshots__/dialog-box-default.snap.png +0 -0
  364. package/src/components/DialogBox/tests/__snapshots__/dialog-box-no-actions.snap.png +0 -0
  365. package/src/components/HeaderBar/HeaderBurgerMenu/tests/__snapshots__/header-burger-menu-generated-submenu-open.snap.png +0 -0
  366. package/src/components/HeaderBar/HeaderBurgerMenu/tests/__snapshots__/header-burger-menu-generated.snap.png +0 -0
  367. package/src/components/HeaderBar/tests/__snapshots__/header-bar-custom-width.snap.png +0 -0
  368. package/src/components/HeaderBar/tests/__snapshots__/header-bar-default.snap.png +0 -0
  369. package/src/components/HeaderBar/tests/__snapshots__/header-bar-no-sticky.snap.png +0 -0
  370. package/src/components/HeaderBar/tests/__snapshots__/header-bar-with-prepend.snap.png +0 -0
  371. package/src/components/HeaderBar/tests/__snapshots__/header-bar-with-side.snap.png +0 -0
  372. package/src/components/HeaderBar/tests/__snapshots__/header-bar-with-subtitle.snap.png +0 -0
  373. package/src/components/Logo/tests/__snapshots__/logo-avatar.snap.png +0 -0
  374. package/src/components/Logo/tests/__snapshots__/logo-dark.snap.png +0 -0
  375. package/src/components/Logo/tests/__snapshots__/logo-default.snap.png +0 -0
  376. package/src/components/Logo/tests/__snapshots__/logo-no-organism.snap.png +0 -0
  377. package/src/components/Logo/tests/__snapshots__/logo-no-signature.snap.png +0 -0
  378. package/src/components/Logo/tests/__snapshots__/logo-risque-pro.snap.png +0 -0
  379. package/src/components/RangeField/tests/__snapshots__/range-field-custom-bg.snap.png +0 -0
  380. package/src/components/RangeField/tests/__snapshots__/range-field-custom-range.snap.png +0 -0
  381. package/src/components/RangeField/tests/__snapshots__/range-field-default.snap.png +0 -0
  382. package/src/components/RangeField/tests/__snapshots__/range-field-step.snap.png +0 -0
  383. package/src/components/RangeField/tests/__snapshots__/range-field-with-label.snap.png +0 -0
  384. package/src/components/SyAlert/tests/__snapshots__/sy-alert-closable.snap.png +0 -0
  385. package/src/components/SyAlert/tests/__snapshots__/sy-alert-error.snap.png +0 -0
  386. package/src/components/SyAlert/tests/__snapshots__/sy-alert-info.snap.png +0 -0
  387. package/src/components/SyAlert/tests/__snapshots__/sy-alert-success.snap.png +0 -0
  388. package/src/components/SyAlert/tests/__snapshots__/sy-alert-variant-outlined.snap.png +0 -0
  389. package/src/components/SyAlert/tests/__snapshots__/sy-alert-variant-tonal.snap.png +0 -0
  390. package/src/components/SyAlert/tests/__snapshots__/sy-alert-warning.snap.png +0 -0
  391. package/src/composables/date/tests/useDateFormat.spec.ts +0 -67
  392. package/src/composables/date/useDateFormat.ts +0 -110
  393. package/src/composables/date/useDateInitialization.ts +0 -92
@@ -0,0 +1,589 @@
1
+ import { mount, flushPromises } from '@vue/test-utils'
2
+ import { describe, it, expect, vi, afterEach } from 'vitest'
3
+ import { defineComponent } from 'vue'
4
+ import { createRouter, createMemoryHistory } from 'vue-router'
5
+ import HorizontalNavbar from '../HorizontalNavbar.vue'
6
+
7
+ const SyTabsStub = defineComponent({
8
+ name: 'SyTabs',
9
+ props: {
10
+ items: { type: Array, default: () => [] },
11
+ modelValue: { type: Number, default: -1 },
12
+ confirmTabChange: { type: Boolean, default: false },
13
+ confirmationMessage: { type: String, default: undefined },
14
+ },
15
+ emits: ['update:modelValue', 'cancel-navigation', 'confirm-tab-change'],
16
+ template: `
17
+ <div class="sy-tabs-stub">
18
+ <button
19
+ v-for="(item, i) in items"
20
+ :key="i"
21
+ :class="['tab-btn', { active: modelValue === i }]"
22
+ :data-index="i"
23
+ @click="$emit('update:modelValue', i)"
24
+ >{{ item.label }}</button>
25
+ </div>
26
+ `,
27
+ })
28
+
29
+ const stubs = {
30
+ RouterLink: true,
31
+ SyTabs: SyTabsStub,
32
+ }
33
+
34
+ const defaultItems = [
35
+ { label: 'Accueil', to: '/' },
36
+ { label: 'À propos', to: '/about' },
37
+ { label: 'Contact', to: '/contact' },
38
+ ]
39
+
40
+ describe('HorizontalNavbar', () => {
41
+ afterEach(() => {
42
+ vi.restoreAllMocks()
43
+ })
44
+
45
+ describe('rendu', () => {
46
+ it('rend le composant avec la classe horizontal-menu', () => {
47
+ const wrapper = mount(HorizontalNavbar, {
48
+ global: { stubs },
49
+ props: { items: defaultItems },
50
+ })
51
+ expect(wrapper.find('.horizontal-menu').exists()).toBe(true)
52
+ })
53
+
54
+ it('rend autant d\'onglets que d\'items', () => {
55
+ const wrapper = mount(HorizontalNavbar, {
56
+ global: { stubs },
57
+ props: { items: defaultItems },
58
+ })
59
+ expect(wrapper.findAll('.tab-btn')).toHaveLength(3)
60
+ })
61
+
62
+ it('rend sans items sans crash', () => {
63
+ const wrapper = mount(HorizontalNavbar, {
64
+ global: { stubs },
65
+ props: { items: [] },
66
+ })
67
+ expect(wrapper.find('.horizontal-menu').exists()).toBe(true)
68
+ expect(wrapper.findAll('.tab-btn')).toHaveLength(0)
69
+ })
70
+
71
+ it('applique la largeur personnalisée via le prop width', () => {
72
+ const wrapper = mount(HorizontalNavbar, {
73
+ global: { stubs },
74
+ props: { items: defaultItems, width: '1200px' },
75
+ })
76
+ expect(wrapper.html()).toBeTruthy()
77
+ })
78
+ })
79
+
80
+ describe('slots', () => {
81
+ it('rend le slot navigation-bar-prepend', () => {
82
+ const wrapper = mount(HorizontalNavbar, {
83
+ global: { stubs },
84
+ props: { items: defaultItems },
85
+ slots: {
86
+ 'navigation-bar-prepend': '<span class="prepend-content">Logo</span>',
87
+ 'navigation-bar-append': '',
88
+ 'default': '',
89
+ },
90
+ })
91
+ expect(wrapper.find('.prepend-content').exists()).toBe(true)
92
+ expect(wrapper.find('.prepend-content').text()).toBe('Logo')
93
+ })
94
+
95
+ it('rend le slot navigation-bar-append', () => {
96
+ const wrapper = mount(HorizontalNavbar, {
97
+ global: { stubs },
98
+ props: { items: defaultItems },
99
+ slots: {
100
+ 'navigation-bar-prepend': '',
101
+ 'navigation-bar-append': '<span class="append-content">Actions</span>',
102
+ 'default': '',
103
+ },
104
+ })
105
+ expect(wrapper.find('.append-content').exists()).toBe(true)
106
+ })
107
+ })
108
+
109
+ describe('handleTabChange', () => {
110
+ it('met à jour activeTab quand un onglet est cliqué', async () => {
111
+ const wrapper = mount(HorizontalNavbar, {
112
+ global: { stubs },
113
+ props: { items: defaultItems },
114
+ })
115
+
116
+ await wrapper.findAll('.tab-btn')[1]!.trigger('click')
117
+ await flushPromises()
118
+
119
+ const vm = wrapper.vm as unknown as { activeTab: number }
120
+ expect(vm.activeTab).toBe(1)
121
+ })
122
+
123
+ it('ne navigue pas via router.push si l\'item est désactivé', async () => {
124
+ const mockPush = vi.fn()
125
+ const items = [
126
+ { label: 'Accueil', to: '/' },
127
+ { label: 'Désactivé', to: '/disabled', disabled: true },
128
+ ]
129
+ const wrapper = mount(HorizontalNavbar, {
130
+ global: {
131
+ stubs,
132
+ mocks: { $router: { push: mockPush, currentRoute: { value: { path: '/' } } } },
133
+ },
134
+ props: { items },
135
+ })
136
+
137
+ await wrapper.findAll('.tab-btn')[1]!.trigger('click')
138
+ await flushPromises()
139
+
140
+ expect(mockPush).not.toHaveBeenCalled()
141
+ })
142
+
143
+ it('émet cancel-navigation depuis SyTabs', async () => {
144
+ const wrapper = mount(HorizontalNavbar, {
145
+ global: { stubs },
146
+ props: { items: defaultItems },
147
+ })
148
+
149
+ wrapper.findComponent(SyTabsStub).vm.$emit('cancel-navigation')
150
+ await flushPromises()
151
+
152
+ expect(wrapper.emitted('cancel-navigation')).toBeTruthy()
153
+ })
154
+ })
155
+
156
+ describe('confirmTabChange', () => {
157
+ it('transmet confirmTabChange=true à SyTabs', () => {
158
+ const wrapper = mount(HorizontalNavbar, {
159
+ global: { stubs },
160
+ props: { items: defaultItems, confirmTabChange: true },
161
+ })
162
+
163
+ const tabs = wrapper.findComponent(SyTabsStub)
164
+ expect(tabs.props('confirmTabChange')).toBe(true)
165
+ })
166
+
167
+ it('transmet un confirmationMessage string à SyTabs', () => {
168
+ const wrapper = mount(HorizontalNavbar, {
169
+ global: { stubs },
170
+ props: {
171
+ items: defaultItems,
172
+ confirmTabChange: true,
173
+ confirmationMessage: 'Voulez-vous continuer ?',
174
+ },
175
+ })
176
+
177
+ const tabs = wrapper.findComponent(SyTabsStub)
178
+ expect(tabs.props('confirmationMessage')).toBe('Voulez-vous continuer ?')
179
+ })
180
+
181
+ it('formattedConfirmationMessage est undefined si confirmationMessage est un booléen', () => {
182
+ const wrapper = mount(HorizontalNavbar, {
183
+ global: { stubs },
184
+ props: {
185
+ items: defaultItems,
186
+ confirmTabChange: true,
187
+ confirmationMessage: true,
188
+ },
189
+ })
190
+
191
+ const tabs = wrapper.findComponent(SyTabsStub)
192
+ expect(tabs.props('confirmationMessage')).toBeUndefined()
193
+ })
194
+
195
+ it('émet confirm-tab-change avec le message et le callback', async () => {
196
+ const wrapper = mount(HorizontalNavbar, {
197
+ global: { stubs },
198
+ props: { items: defaultItems, confirmTabChange: true },
199
+ })
200
+
201
+ const callback = vi.fn()
202
+ wrapper.findComponent(SyTabsStub).vm.$emit('confirm-tab-change', 'Message', callback)
203
+ await flushPromises()
204
+
205
+ expect(wrapper.emitted('confirm-tab-change')).toBeTruthy()
206
+ const [msg, cb] = wrapper.emitted('confirm-tab-change')![0] as [string, () => void]
207
+ expect(msg).toBe('Message')
208
+ expect(cb).toBe(callback)
209
+ })
210
+ })
211
+
212
+ describe('resetTabSelection', () => {
213
+ it('retourne activeTab=-1 et activeItemIndex=-1 si items est vide', () => {
214
+ const wrapper = mount(HorizontalNavbar, {
215
+ global: { stubs },
216
+ props: { items: [] },
217
+ })
218
+
219
+ const vm = wrapper.vm as unknown as { resetTabSelection: () => { activeTab: number, activeItemIndex: number } }
220
+ const result = vm.resetTabSelection()
221
+ expect(result.activeTab).toBe(-1)
222
+ expect(result.activeItemIndex).toBe(-1)
223
+ })
224
+
225
+ it('est exposée et appelable depuis l\'extérieur', () => {
226
+ const wrapper = mount(HorizontalNavbar, {
227
+ global: { stubs },
228
+ props: { items: defaultItems },
229
+ })
230
+
231
+ const vm = wrapper.vm as unknown as { resetTabSelection: () => unknown }
232
+ expect(typeof vm.resetTabSelection).toBe('function')
233
+ expect(() => vm.resetTabSelection()).not.toThrow()
234
+ })
235
+ })
236
+
237
+ describe('isActive', () => {
238
+ it('retourne false pour un item désactivé', () => {
239
+ const wrapper = mount(HorizontalNavbar, {
240
+ global: { stubs },
241
+ props: {
242
+ items: [{ label: 'Test', to: '/', disabled: true }],
243
+ },
244
+ })
245
+
246
+ const vm = wrapper.vm as unknown as {
247
+ isActive: (item: { label: string, to: string, disabled?: boolean }, index: number) => boolean
248
+ }
249
+ expect(vm.isActive({ label: 'Test', to: '/', disabled: true }, 0)).toBe(false)
250
+ })
251
+
252
+ it('retourne false pour un item avec to (string) quand pathname ne correspond pas', () => {
253
+ const wrapper = mount(HorizontalNavbar, {
254
+ global: { stubs },
255
+ props: { items: [{ label: 'À propos', to: '/about' }] },
256
+ })
257
+
258
+ const vm = wrapper.vm as unknown as {
259
+ isActive: (item: { label: string, to: string }, index: number) => boolean
260
+ }
261
+ // jsdom pathname est '/', '/about' ne correspond pas
262
+ expect(vm.isActive({ label: 'À propos', to: '/about' }, 0)).toBe(false)
263
+ })
264
+
265
+ it('retourne true pour un item avec to (string) correspondant à window.location.pathname', () => {
266
+ Object.defineProperty(window, 'location', {
267
+ value: { ...window.location, pathname: '/about', href: 'http://localhost/about' },
268
+ writable: true,
269
+ })
270
+ const wrapper = mount(HorizontalNavbar, {
271
+ global: { stubs },
272
+ props: { items: [{ label: 'À propos', to: '/about' }] },
273
+ })
274
+
275
+ const vm = wrapper.vm as unknown as {
276
+ isActive: (item: { label: string, to: string }, index: number) => boolean
277
+ }
278
+ expect(vm.isActive({ label: 'À propos', to: '/about' }, 0)).toBe(true)
279
+ Object.defineProperty(window, 'location', {
280
+ value: { ...window.location, pathname: '/', href: 'http://localhost/' },
281
+ writable: true,
282
+ })
283
+ })
284
+
285
+ it('retourne true pour un item avec to (string) sous-chemin de pathname', () => {
286
+ Object.defineProperty(window, 'location', {
287
+ value: { ...window.location, pathname: '/about/team', href: 'http://localhost/about/team' },
288
+ writable: true,
289
+ })
290
+ const wrapper = mount(HorizontalNavbar, {
291
+ global: { stubs },
292
+ props: { items: [{ label: 'À propos', to: '/about' }] },
293
+ })
294
+
295
+ const vm = wrapper.vm as unknown as {
296
+ isActive: (item: { label: string, to: string }, index: number) => boolean
297
+ }
298
+ expect(vm.isActive({ label: 'À propos', to: '/about' }, 0)).toBe(true)
299
+ Object.defineProperty(window, 'location', {
300
+ value: { ...window.location, pathname: '/', href: 'http://localhost/' },
301
+ writable: true,
302
+ })
303
+ })
304
+
305
+ it('retourne true pour un item avec to (objet) correspondant à pathname', () => {
306
+ Object.defineProperty(window, 'location', {
307
+ value: { ...window.location, pathname: '/contact', href: 'http://localhost/contact' },
308
+ writable: true,
309
+ })
310
+ const wrapper = mount(HorizontalNavbar, {
311
+ global: { stubs },
312
+ props: { items: [{ label: 'Contact', to: { path: '/contact' } }] },
313
+ })
314
+
315
+ const vm = wrapper.vm as unknown as {
316
+ isActive: (item: { label: string, to: { path: string } }, index: number) => boolean
317
+ }
318
+ expect(vm.isActive({ label: 'Contact', to: { path: '/contact' } }, 0)).toBe(true)
319
+ Object.defineProperty(window, 'location', {
320
+ value: { ...window.location, pathname: '/', href: 'http://localhost/' },
321
+ writable: true,
322
+ })
323
+ })
324
+
325
+ it('retourne false pour un item avec to (objet) non correspondant', () => {
326
+ const wrapper = mount(HorizontalNavbar, {
327
+ global: { stubs },
328
+ props: { items: [{ label: 'Contact', to: { path: '/contact' } }] },
329
+ })
330
+
331
+ const vm = wrapper.vm as unknown as {
332
+ isActive: (item: { label: string, to: { path: string } }, index: number) => boolean
333
+ }
334
+ // jsdom pathname est '/', '/contact' ne correspond pas
335
+ expect(vm.isActive({ label: 'Contact', to: { path: '/contact' } }, 0)).toBe(false)
336
+ })
337
+ })
338
+
339
+ describe('handleTabChange — navigation', () => {
340
+ it('met à jour activeTab au clic sur un onglet (sans router réel)', async () => {
341
+ const wrapper = mount(HorizontalNavbar, {
342
+ global: { stubs },
343
+ props: { items: defaultItems },
344
+ })
345
+
346
+ await wrapper.findAll('.tab-btn')[1]!.trigger('click')
347
+ await flushPromises()
348
+
349
+ const vm = wrapper.vm as unknown as { activeTab: number }
350
+ expect(vm.activeTab).toBe(1)
351
+ })
352
+
353
+ it('met à jour activeTab mais retourne tôt si confirmTabChange est true', async () => {
354
+ const wrapper = mount(HorizontalNavbar, {
355
+ global: { stubs },
356
+ props: { items: defaultItems, confirmTabChange: true },
357
+ })
358
+
359
+ await wrapper.findAll('.tab-btn')[1]!.trigger('click')
360
+ await flushPromises()
361
+
362
+ const vm = wrapper.vm as unknown as { activeTab: number }
363
+ expect(vm.activeTab).toBe(1)
364
+ })
365
+
366
+ it('navigue via window.location.href si item a un href', async () => {
367
+ const locationSpy = vi.spyOn(window, 'location', 'get').mockReturnValue({
368
+ ...window.location,
369
+ href: '',
370
+ } as Location)
371
+ const hrefSetter = vi.fn()
372
+ Object.defineProperty(window, 'location', {
373
+ value: { ...window.location, set href(v: string) { hrefSetter(v) } },
374
+ writable: true,
375
+ })
376
+
377
+ const items = [{ label: 'Externe', href: 'https://example.com' }]
378
+ const wrapper = mount(HorizontalNavbar, {
379
+ global: { stubs },
380
+ props: { items },
381
+ })
382
+
383
+ await wrapper.findAll('.tab-btn')[0]!.trigger('click')
384
+ await flushPromises()
385
+
386
+ locationSpy.mockRestore()
387
+ })
388
+
389
+ it('ne plante pas si l\'item cliqué n\'existe pas dans items', async () => {
390
+ const wrapper = mount(HorizontalNavbar, {
391
+ global: { stubs },
392
+ props: { items: defaultItems },
393
+ })
394
+
395
+ const vm = wrapper.vm as unknown as { handleTabChange: (index: number) => Promise<void> }
396
+ await expect(vm.handleTabChange(999)).resolves.toBeUndefined()
397
+ })
398
+ })
399
+
400
+ describe('resetTabSelection — avec items et route', () => {
401
+ it('retourne les valeurs courantes de activeTab et activeItemIndex', () => {
402
+ const wrapper = mount(HorizontalNavbar, {
403
+ global: { stubs },
404
+ props: { items: defaultItems },
405
+ })
406
+
407
+ const vm = wrapper.vm as unknown as {
408
+ resetTabSelection: () => { activeTab: number, activeItemIndex: number }
409
+ }
410
+ const result = vm.resetTabSelection()
411
+ expect(typeof result.activeTab).toBe('number')
412
+ expect(typeof result.activeItemIndex).toBe('number')
413
+ })
414
+
415
+ it('retourne -1 si aucun item ne correspond à la route', () => {
416
+ const wrapper = mount(HorizontalNavbar, {
417
+ global: { stubs },
418
+ props: { items: defaultItems },
419
+ })
420
+
421
+ const vm = wrapper.vm as unknown as {
422
+ resetTabSelection: () => { activeTab: number, activeItemIndex: number }
423
+ }
424
+ const result = vm.resetTabSelection()
425
+ expect(result.activeTab).toBe(-1)
426
+ })
427
+
428
+ it('met à jour activeTab après avoir appelé handleTabChange puis resetTabSelection', async () => {
429
+ const wrapper = mount(HorizontalNavbar, {
430
+ global: { stubs },
431
+ props: { items: defaultItems },
432
+ })
433
+
434
+ await wrapper.findAll('.tab-btn')[1]!.trigger('click')
435
+ await flushPromises()
436
+
437
+ const vm = wrapper.vm as unknown as {
438
+ activeTab: number
439
+ resetTabSelection: () => { activeTab: number, activeItemIndex: number }
440
+ }
441
+ expect(vm.activeTab).toBe(1)
442
+ })
443
+ })
444
+
445
+ describe('avec vue-router réel', () => {
446
+ function makeRouter() {
447
+ return createRouter({
448
+ history: createMemoryHistory(),
449
+ routes: [
450
+ { path: '/', component: { template: '<div/>' } },
451
+ { path: '/about', component: { template: '<div/>' } },
452
+ { path: '/contact', component: { template: '<div/>' } },
453
+ { path: '/about/team', component: { template: '<div/>' } },
454
+ ],
455
+ })
456
+ }
457
+
458
+ it('isActive retourne true quand la route correspond à un item to (string)', async () => {
459
+ const router = makeRouter()
460
+ await router.push('/about')
461
+ const wrapper = mount(HorizontalNavbar, {
462
+ global: { stubs, plugins: [router] },
463
+ props: { items: defaultItems },
464
+ })
465
+ await flushPromises()
466
+
467
+ const vm = wrapper.vm as unknown as {
468
+ isActive: (item: { label: string, to: string }, index: number) => boolean
469
+ }
470
+ expect(vm.isActive({ label: 'À propos', to: '/about' }, 1)).toBe(true)
471
+ })
472
+
473
+ it('resetTabSelection trouve l\'item actif avec une route correspondante', async () => {
474
+ const router = makeRouter()
475
+ await router.push('/about')
476
+ const wrapper = mount(HorizontalNavbar, {
477
+ global: { stubs, plugins: [router] },
478
+ props: { items: defaultItems },
479
+ })
480
+ await flushPromises()
481
+
482
+ const vm = wrapper.vm as unknown as {
483
+ resetTabSelection: () => { activeTab: number, activeItemIndex: number }
484
+ }
485
+ const result = vm.resetTabSelection()
486
+ expect(result.activeTab).toBe(1)
487
+ expect(result.activeItemIndex).toBe(1)
488
+ })
489
+
490
+ it('watcher currentPath met à jour activeTab lors d\'un changement de route', async () => {
491
+ const router = makeRouter()
492
+ await router.push('/')
493
+ const wrapper = mount(HorizontalNavbar, {
494
+ global: { stubs, plugins: [router] },
495
+ props: { items: defaultItems },
496
+ })
497
+ await flushPromises()
498
+
499
+ await router.push('/about')
500
+ await flushPromises()
501
+
502
+ const vm = wrapper.vm as unknown as { activeTab: number }
503
+ expect(vm.activeTab).toBe(1)
504
+ })
505
+
506
+ it('watcher currentPath remet activeTab à -1 si aucun item ne correspond', async () => {
507
+ const router = makeRouter()
508
+ await router.push('/about')
509
+ const itemsWithoutContact = [
510
+ { label: 'Accueil', to: '/home' },
511
+ { label: 'À propos', to: '/about' },
512
+ ]
513
+ const wrapper = mount(HorizontalNavbar, {
514
+ global: { stubs, plugins: [router] },
515
+ props: { items: itemsWithoutContact },
516
+ })
517
+ await flushPromises()
518
+
519
+ await router.push('/contact')
520
+ await flushPromises()
521
+
522
+ const vm = wrapper.vm as unknown as { activeTab: number }
523
+ expect(vm.activeTab).toBe(-1)
524
+ })
525
+
526
+ it('handleTabChange appelle router.push avec le to de l\'item', async () => {
527
+ const router = makeRouter()
528
+ const pushSpy = vi.spyOn(router, 'push')
529
+ await router.push('/')
530
+ const wrapper = mount(HorizontalNavbar, {
531
+ global: { stubs, plugins: [router] },
532
+ props: { items: defaultItems },
533
+ })
534
+ await flushPromises()
535
+
536
+ await wrapper.findAll('.tab-btn')[1]!.trigger('click')
537
+ await flushPromises()
538
+
539
+ expect(pushSpy).toHaveBeenCalledWith('/about')
540
+ })
541
+
542
+ it('isActive retourne true pour un item avec to (string) sous-chemin', async () => {
543
+ const router = makeRouter()
544
+ await router.push('/about/team')
545
+ const wrapper = mount(HorizontalNavbar, {
546
+ global: { stubs, plugins: [router] },
547
+ props: { items: defaultItems },
548
+ })
549
+ await flushPromises()
550
+
551
+ const vm = wrapper.vm as unknown as {
552
+ isActive: (item: { label: string, to: string }, index: number) => boolean
553
+ }
554
+ expect(vm.isActive({ label: 'À propos', to: '/about' }, 1)).toBe(true)
555
+ })
556
+ })
557
+
558
+ describe('tabItems computed', () => {
559
+ it('convertit les items en TabItem avec les bonnes propriétés', () => {
560
+ const wrapper = mount(HorizontalNavbar, {
561
+ global: { stubs },
562
+ props: {
563
+ items: [
564
+ { label: 'Accueil', to: '/', disabled: false },
565
+ { label: 'Externe', href: 'https://example.com' },
566
+ ],
567
+ },
568
+ })
569
+
570
+ const vm = wrapper.vm as unknown as {
571
+ tabItems: Array<{ label: string, value: number, href?: string, to?: string, disabled?: boolean }>
572
+ }
573
+ expect(vm.tabItems).toHaveLength(2)
574
+ expect(vm.tabItems[0]!.label).toBe('Accueil')
575
+ expect(vm.tabItems[0]!.value).toBe(0)
576
+ expect(vm.tabItems[1]!.href).toBe('https://example.com')
577
+ })
578
+
579
+ it('retourne un tableau vide si items n\'est pas un tableau', () => {
580
+ const wrapper = mount(HorizontalNavbar, {
581
+ global: { stubs },
582
+ props: { items: [] },
583
+ })
584
+
585
+ const vm = wrapper.vm as unknown as { tabItems: unknown[] }
586
+ expect(vm.tabItems).toHaveLength(0)
587
+ })
588
+ })
589
+ })