shadcn-rails 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (315) hide show
  1. checksums.yaml +7 -0
  2. data/.dockerignore +40 -0
  3. data/CHANGELOG.md +54 -0
  4. data/CLAUDE.md +463 -0
  5. data/PROGRESS.md +485 -0
  6. data/README.md +1483 -0
  7. data/Rakefile +29 -0
  8. data/__tests__/controllers/__snapshots__/calendar_controller.test.js.snap +13 -0
  9. data/__tests__/controllers/__snapshots__/popover_controller.test.js.snap +46 -0
  10. data/__tests__/controllers/__snapshots__/sheet_controller.test.js.snap +111 -0
  11. data/__tests__/controllers/__snapshots__/tabs_controller.test.js.snap +27 -0
  12. data/__tests__/controllers/accordion_controller.test.js +904 -0
  13. data/__tests__/controllers/calendar_controller.test.js +1370 -0
  14. data/__tests__/controllers/carousel_controller.test.js +912 -0
  15. data/__tests__/controllers/checkbox_controller.test.js +454 -0
  16. data/__tests__/controllers/collapsible_controller.test.js +407 -0
  17. data/__tests__/controllers/combobox_controller.test.js +966 -0
  18. data/__tests__/controllers/context_menu_controller.test.js +627 -0
  19. data/__tests__/controllers/date_picker_controller.test.js +636 -0
  20. data/__tests__/controllers/dialog_controller.test.js +878 -0
  21. data/__tests__/controllers/drawer_controller.test.js +995 -0
  22. data/__tests__/controllers/menubar_controller.test.js +736 -0
  23. data/__tests__/controllers/navigation_menu_controller.test.js +598 -0
  24. data/__tests__/controllers/popover_controller.test.js +1007 -0
  25. data/__tests__/controllers/radio_group_controller.test.js +640 -0
  26. data/__tests__/controllers/resizable_controller.test.js +680 -0
  27. data/__tests__/controllers/select_controller.test.js +674 -0
  28. data/__tests__/controllers/sheet_controller.test.js +986 -0
  29. data/__tests__/controllers/slider_controller.test.js +1036 -0
  30. data/__tests__/controllers/switch_controller.test.js +424 -0
  31. data/__tests__/controllers/tabs_controller.test.js +907 -0
  32. data/__tests__/controllers/toggle_group_controller.test.js +839 -0
  33. data/__tests__/controllers/tooltip_controller.test.js +808 -0
  34. data/__tests__/helpers/stimulus-test-helper.js +203 -0
  35. data/app/assets/config/manifest.js +1 -0
  36. data/app/assets/javascripts/shadcn/controllers/accordion_controller.d.ts +53 -0
  37. data/app/assets/javascripts/shadcn/controllers/accordion_controller.js +140 -0
  38. data/app/assets/javascripts/shadcn/controllers/avatar_controller.d.ts +22 -0
  39. data/app/assets/javascripts/shadcn/controllers/avatar_controller.js +26 -0
  40. data/app/assets/javascripts/shadcn/controllers/calendar_controller.js +592 -0
  41. data/app/assets/javascripts/shadcn/controllers/carousel_controller.js +263 -0
  42. data/app/assets/javascripts/shadcn/controllers/checkbox_controller.d.ts +31 -0
  43. data/app/assets/javascripts/shadcn/controllers/checkbox_controller.js +48 -0
  44. data/app/assets/javascripts/shadcn/controllers/collapsible_controller.d.ts +43 -0
  45. data/app/assets/javascripts/shadcn/controllers/collapsible_controller.js +73 -0
  46. data/app/assets/javascripts/shadcn/controllers/combobox_controller.js +234 -0
  47. data/app/assets/javascripts/shadcn/controllers/command_controller.js +141 -0
  48. data/app/assets/javascripts/shadcn/controllers/command_dialog_controller.js +162 -0
  49. data/app/assets/javascripts/shadcn/controllers/context_menu_controller.js +202 -0
  50. data/app/assets/javascripts/shadcn/controllers/date_picker_controller.js +282 -0
  51. data/app/assets/javascripts/shadcn/controllers/dialog_controller.d.ts +67 -0
  52. data/app/assets/javascripts/shadcn/controllers/dialog_controller.js +187 -0
  53. data/app/assets/javascripts/shadcn/controllers/drawer_controller.d.ts +58 -0
  54. data/app/assets/javascripts/shadcn/controllers/drawer_controller.js +112 -0
  55. data/app/assets/javascripts/shadcn/controllers/dropdown_controller.d.ts +83 -0
  56. data/app/assets/javascripts/shadcn/controllers/dropdown_controller.js +225 -0
  57. data/app/assets/javascripts/shadcn/controllers/hover_card_controller.d.ts +59 -0
  58. data/app/assets/javascripts/shadcn/controllers/hover_card_controller.js +143 -0
  59. data/app/assets/javascripts/shadcn/controllers/input_otp_controller.d.ts +44 -0
  60. data/app/assets/javascripts/shadcn/controllers/input_otp_controller.js +206 -0
  61. data/app/assets/javascripts/shadcn/controllers/menubar_controller.js +323 -0
  62. data/app/assets/javascripts/shadcn/controllers/navigation_menu_controller.js +251 -0
  63. data/app/assets/javascripts/shadcn/controllers/popover_controller.d.ts +56 -0
  64. data/app/assets/javascripts/shadcn/controllers/popover_controller.js +141 -0
  65. data/app/assets/javascripts/shadcn/controllers/radio_group_controller.d.ts +47 -0
  66. data/app/assets/javascripts/shadcn/controllers/radio_group_controller.js +108 -0
  67. data/app/assets/javascripts/shadcn/controllers/resizable_controller.js +272 -0
  68. data/app/assets/javascripts/shadcn/controllers/scroll_area_controller.d.ts +44 -0
  69. data/app/assets/javascripts/shadcn/controllers/scroll_area_controller.js +74 -0
  70. data/app/assets/javascripts/shadcn/controllers/select_controller.d.ts +84 -0
  71. data/app/assets/javascripts/shadcn/controllers/select_controller.js +222 -0
  72. data/app/assets/javascripts/shadcn/controllers/sheet_controller.d.ts +60 -0
  73. data/app/assets/javascripts/shadcn/controllers/sheet_controller.js +151 -0
  74. data/app/assets/javascripts/shadcn/controllers/sidebar_controller.js +148 -0
  75. data/app/assets/javascripts/shadcn/controllers/slider_controller.d.ts +102 -0
  76. data/app/assets/javascripts/shadcn/controllers/slider_controller.js +364 -0
  77. data/app/assets/javascripts/shadcn/controllers/switch_controller.d.ts +46 -0
  78. data/app/assets/javascripts/shadcn/controllers/switch_controller.js +78 -0
  79. data/app/assets/javascripts/shadcn/controllers/tabs_controller.d.ts +51 -0
  80. data/app/assets/javascripts/shadcn/controllers/tabs_controller.js +126 -0
  81. data/app/assets/javascripts/shadcn/controllers/toast_controller.d.ts +37 -0
  82. data/app/assets/javascripts/shadcn/controllers/toast_controller.js +58 -0
  83. data/app/assets/javascripts/shadcn/controllers/toggle_controller.d.ts +27 -0
  84. data/app/assets/javascripts/shadcn/controllers/toggle_controller.js +42 -0
  85. data/app/assets/javascripts/shadcn/controllers/toggle_group_controller.d.ts +44 -0
  86. data/app/assets/javascripts/shadcn/controllers/toggle_group_controller.js +68 -0
  87. data/app/assets/javascripts/shadcn/controllers/tooltip_controller.d.ts +56 -0
  88. data/app/assets/javascripts/shadcn/controllers/tooltip_controller.js +117 -0
  89. data/app/assets/javascripts/shadcn/index.d.ts +74 -0
  90. data/app/assets/javascripts/shadcn/index.js +133 -0
  91. data/app/assets/stylesheets/.keep +0 -0
  92. data/app/assets/stylesheets/shadcn/base.css +445 -0
  93. data/app/assets/stylesheets/shadcn/components.css +513 -0
  94. data/app/assets/stylesheets/shadcn/index.css +18 -0
  95. data/app/assets/stylesheets/shadcn/themes/gray.css +68 -0
  96. data/app/assets/stylesheets/shadcn/themes/slate.css +68 -0
  97. data/app/assets/stylesheets/shadcn/themes/stone.css +68 -0
  98. data/app/assets/stylesheets/shadcn/themes/zinc.css +68 -0
  99. data/app/components/shadcn/accordion_component.rb +63 -0
  100. data/app/components/shadcn/accordion_content_component.rb +29 -0
  101. data/app/components/shadcn/accordion_item_component.rb +40 -0
  102. data/app/components/shadcn/accordion_trigger_component.rb +49 -0
  103. data/app/components/shadcn/alert_component.rb +75 -0
  104. data/app/components/shadcn/alert_description_component.rb +12 -0
  105. data/app/components/shadcn/alert_dialog_action_component.rb +24 -0
  106. data/app/components/shadcn/alert_dialog_cancel_component.rb +24 -0
  107. data/app/components/shadcn/alert_dialog_component.rb +71 -0
  108. data/app/components/shadcn/alert_dialog_content_component.rb +57 -0
  109. data/app/components/shadcn/alert_dialog_description_component.rb +12 -0
  110. data/app/components/shadcn/alert_dialog_footer_component.rb +19 -0
  111. data/app/components/shadcn/alert_dialog_header_component.rb +19 -0
  112. data/app/components/shadcn/alert_dialog_title_component.rb +12 -0
  113. data/app/components/shadcn/alert_title_component.rb +12 -0
  114. data/app/components/shadcn/aspect_ratio_component.rb +49 -0
  115. data/app/components/shadcn/avatar_component.rb +107 -0
  116. data/app/components/shadcn/avatar_fallback_component.rb +17 -0
  117. data/app/components/shadcn/badge_component.rb +49 -0
  118. data/app/components/shadcn/base_component.rb +100 -0
  119. data/app/components/shadcn/breadcrumb_component.rb +70 -0
  120. data/app/components/shadcn/breadcrumb_item_component.rb +50 -0
  121. data/app/components/shadcn/button_component.rb +141 -0
  122. data/app/components/shadcn/button_group_component.rb +69 -0
  123. data/app/components/shadcn/calendar_component.rb +337 -0
  124. data/app/components/shadcn/card_action_component.rb +10 -0
  125. data/app/components/shadcn/card_component.rb +63 -0
  126. data/app/components/shadcn/card_content_component.rb +19 -0
  127. data/app/components/shadcn/card_description_component.rb +12 -0
  128. data/app/components/shadcn/card_footer_component.rb +12 -0
  129. data/app/components/shadcn/card_header_component.rb +24 -0
  130. data/app/components/shadcn/card_title_component.rb +18 -0
  131. data/app/components/shadcn/carousel_component.rb +275 -0
  132. data/app/components/shadcn/checkbox_component.rb +103 -0
  133. data/app/components/shadcn/collapsible_component.rb +66 -0
  134. data/app/components/shadcn/collapsible_content_component.rb +28 -0
  135. data/app/components/shadcn/combobox_component.rb +322 -0
  136. data/app/components/shadcn/command_component.rb +52 -0
  137. data/app/components/shadcn/command_dialog_component.rb +76 -0
  138. data/app/components/shadcn/command_empty_component.rb +12 -0
  139. data/app/components/shadcn/command_group_component.rb +34 -0
  140. data/app/components/shadcn/command_input_component.rb +59 -0
  141. data/app/components/shadcn/command_item_component.rb +48 -0
  142. data/app/components/shadcn/command_list_component.rb +38 -0
  143. data/app/components/shadcn/command_separator_component.rb +12 -0
  144. data/app/components/shadcn/command_shortcut_component.rb +12 -0
  145. data/app/components/shadcn/context_menu_component.rb +64 -0
  146. data/app/components/shadcn/context_menu_content_component.rb +44 -0
  147. data/app/components/shadcn/context_menu_item_component.rb +63 -0
  148. data/app/components/shadcn/context_menu_label_component.rb +18 -0
  149. data/app/components/shadcn/context_menu_separator_component.rb +12 -0
  150. data/app/components/shadcn/context_menu_shortcut_component.rb +12 -0
  151. data/app/components/shadcn/date_picker_component.rb +368 -0
  152. data/app/components/shadcn/dialog_component.rb +77 -0
  153. data/app/components/shadcn/dialog_content_component.rb +91 -0
  154. data/app/components/shadcn/dialog_description_component.rb +12 -0
  155. data/app/components/shadcn/dialog_footer_component.rb +12 -0
  156. data/app/components/shadcn/dialog_header_component.rb +19 -0
  157. data/app/components/shadcn/dialog_title_component.rb +12 -0
  158. data/app/components/shadcn/drawer_component.rb +72 -0
  159. data/app/components/shadcn/drawer_content_component.rb +76 -0
  160. data/app/components/shadcn/drawer_description_component.rb +12 -0
  161. data/app/components/shadcn/drawer_footer_component.rb +12 -0
  162. data/app/components/shadcn/drawer_header_component.rb +19 -0
  163. data/app/components/shadcn/drawer_title_component.rb +12 -0
  164. data/app/components/shadcn/dropdown_menu_component.rb +75 -0
  165. data/app/components/shadcn/dropdown_menu_content_component.rb +49 -0
  166. data/app/components/shadcn/dropdown_menu_group_component.rb +10 -0
  167. data/app/components/shadcn/dropdown_menu_item_component.rb +63 -0
  168. data/app/components/shadcn/dropdown_menu_label_component.rb +18 -0
  169. data/app/components/shadcn/dropdown_menu_separator_component.rb +12 -0
  170. data/app/components/shadcn/dropdown_menu_shortcut_component.rb +12 -0
  171. data/app/components/shadcn/empty_component.rb +48 -0
  172. data/app/components/shadcn/empty_content_component.rb +12 -0
  173. data/app/components/shadcn/empty_description_component.rb +12 -0
  174. data/app/components/shadcn/empty_header_component.rb +29 -0
  175. data/app/components/shadcn/empty_media_component.rb +21 -0
  176. data/app/components/shadcn/empty_title_component.rb +12 -0
  177. data/app/components/shadcn/field_component.rb +113 -0
  178. data/app/components/shadcn/hover_card_component.rb +64 -0
  179. data/app/components/shadcn/hover_card_content_component.rb +36 -0
  180. data/app/components/shadcn/input_component.rb +108 -0
  181. data/app/components/shadcn/input_group_component.rb +70 -0
  182. data/app/components/shadcn/input_otp_component.rb +183 -0
  183. data/app/components/shadcn/item_actions_component.rb +12 -0
  184. data/app/components/shadcn/item_component.rb +98 -0
  185. data/app/components/shadcn/item_content_component.rb +24 -0
  186. data/app/components/shadcn/item_description_component.rb +12 -0
  187. data/app/components/shadcn/item_footer_component.rb +12 -0
  188. data/app/components/shadcn/item_group_component.rb +24 -0
  189. data/app/components/shadcn/item_header_component.rb +12 -0
  190. data/app/components/shadcn/item_media_component.rb +22 -0
  191. data/app/components/shadcn/item_separator_component.rb +12 -0
  192. data/app/components/shadcn/item_title_component.rb +12 -0
  193. data/app/components/shadcn/kbd_component.rb +36 -0
  194. data/app/components/shadcn/label_component.rb +49 -0
  195. data/app/components/shadcn/menubar_checkbox_item_component.rb +76 -0
  196. data/app/components/shadcn/menubar_component.rb +56 -0
  197. data/app/components/shadcn/menubar_content_component.rb +64 -0
  198. data/app/components/shadcn/menubar_item_component.rb +65 -0
  199. data/app/components/shadcn/menubar_label_component.rb +27 -0
  200. data/app/components/shadcn/menubar_menu_component.rb +34 -0
  201. data/app/components/shadcn/menubar_radio_group_component.rb +42 -0
  202. data/app/components/shadcn/menubar_radio_item_component.rb +76 -0
  203. data/app/components/shadcn/menubar_separator_component.rb +22 -0
  204. data/app/components/shadcn/menubar_shortcut_component.rb +21 -0
  205. data/app/components/shadcn/menubar_sub_component.rb +38 -0
  206. data/app/components/shadcn/menubar_sub_content_component.rb +45 -0
  207. data/app/components/shadcn/menubar_sub_trigger_component.rb +59 -0
  208. data/app/components/shadcn/menubar_trigger_component.rb +31 -0
  209. data/app/components/shadcn/native_select_component.rb +150 -0
  210. data/app/components/shadcn/navigation_menu_component.rb +76 -0
  211. data/app/components/shadcn/navigation_menu_content_component.rb +30 -0
  212. data/app/components/shadcn/navigation_menu_item_component.rb +39 -0
  213. data/app/components/shadcn/navigation_menu_link_component.rb +38 -0
  214. data/app/components/shadcn/navigation_menu_list_component.rb +29 -0
  215. data/app/components/shadcn/navigation_menu_trigger_component.rb +59 -0
  216. data/app/components/shadcn/pagination_component.rb +195 -0
  217. data/app/components/shadcn/pagination_content_component.rb +47 -0
  218. data/app/components/shadcn/pagination_ellipsis_component.rb +30 -0
  219. data/app/components/shadcn/pagination_item_component.rb +53 -0
  220. data/app/components/shadcn/pagination_next_component.rb +48 -0
  221. data/app/components/shadcn/pagination_previous_component.rb +48 -0
  222. data/app/components/shadcn/popover_component.rb +76 -0
  223. data/app/components/shadcn/popover_content_component.rb +25 -0
  224. data/app/components/shadcn/progress_component.rb +77 -0
  225. data/app/components/shadcn/radio_group_component.rb +129 -0
  226. data/app/components/shadcn/radio_group_item_component.rb +109 -0
  227. data/app/components/shadcn/resizable_handle_component.rb +98 -0
  228. data/app/components/shadcn/resizable_panel_component.rb +56 -0
  229. data/app/components/shadcn/resizable_panel_group_component.rb +94 -0
  230. data/app/components/shadcn/scroll_area_component.rb +110 -0
  231. data/app/components/shadcn/select_component.rb +151 -0
  232. data/app/components/shadcn/select_group_component.rb +32 -0
  233. data/app/components/shadcn/select_item_component.rb +59 -0
  234. data/app/components/shadcn/select_separator_component.rb +12 -0
  235. data/app/components/shadcn/separator_component.rb +54 -0
  236. data/app/components/shadcn/sheet_component.rb +82 -0
  237. data/app/components/shadcn/sheet_content_component.rb +95 -0
  238. data/app/components/shadcn/sheet_description_component.rb +12 -0
  239. data/app/components/shadcn/sheet_footer_component.rb +12 -0
  240. data/app/components/shadcn/sheet_header_component.rb +19 -0
  241. data/app/components/shadcn/sheet_title_component.rb +12 -0
  242. data/app/components/shadcn/sidebar_component.rb +180 -0
  243. data/app/components/shadcn/sidebar_content_component.rb +32 -0
  244. data/app/components/shadcn/sidebar_footer_component.rb +24 -0
  245. data/app/components/shadcn/sidebar_group_action_component.rb +26 -0
  246. data/app/components/shadcn/sidebar_group_component.rb +38 -0
  247. data/app/components/shadcn/sidebar_group_content_component.rb +32 -0
  248. data/app/components/shadcn/sidebar_group_label_component.rb +25 -0
  249. data/app/components/shadcn/sidebar_header_component.rb +24 -0
  250. data/app/components/shadcn/sidebar_inset_component.rb +25 -0
  251. data/app/components/shadcn/sidebar_menu_action_component.rb +37 -0
  252. data/app/components/shadcn/sidebar_menu_badge_component.rb +25 -0
  253. data/app/components/shadcn/sidebar_menu_button_component.rb +52 -0
  254. data/app/components/shadcn/sidebar_menu_component.rb +32 -0
  255. data/app/components/shadcn/sidebar_menu_item_component.rb +41 -0
  256. data/app/components/shadcn/sidebar_menu_skeleton_component.rb +46 -0
  257. data/app/components/shadcn/sidebar_menu_sub_button_component.rb +43 -0
  258. data/app/components/shadcn/sidebar_menu_sub_component.rb +33 -0
  259. data/app/components/shadcn/sidebar_menu_sub_item_component.rb +30 -0
  260. data/app/components/shadcn/sidebar_provider_component.rb +57 -0
  261. data/app/components/shadcn/sidebar_rail_component.rb +30 -0
  262. data/app/components/shadcn/sidebar_separator_component.rb +24 -0
  263. data/app/components/shadcn/sidebar_trigger_component.rb +51 -0
  264. data/app/components/shadcn/skeleton_component.rb +29 -0
  265. data/app/components/shadcn/slider_component.rb +76 -0
  266. data/app/components/shadcn/spinner_component.rb +67 -0
  267. data/app/components/shadcn/switch_component.rb +147 -0
  268. data/app/components/shadcn/table_body_component.rb +16 -0
  269. data/app/components/shadcn/table_caption_component.rb +12 -0
  270. data/app/components/shadcn/table_cell_component.rb +12 -0
  271. data/app/components/shadcn/table_component.rb +57 -0
  272. data/app/components/shadcn/table_footer_component.rb +16 -0
  273. data/app/components/shadcn/table_head_component.rb +12 -0
  274. data/app/components/shadcn/table_header_component.rb +16 -0
  275. data/app/components/shadcn/table_row_component.rb +40 -0
  276. data/app/components/shadcn/tabs_component.rb +78 -0
  277. data/app/components/shadcn/tabs_content_component.rb +32 -0
  278. data/app/components/shadcn/tabs_list_component.rb +30 -0
  279. data/app/components/shadcn/tabs_trigger_component.rb +37 -0
  280. data/app/components/shadcn/textarea_component.rb +84 -0
  281. data/app/components/shadcn/toast_action_component.rb +18 -0
  282. data/app/components/shadcn/toast_component.rb +114 -0
  283. data/app/components/shadcn/toast_description_component.rb +12 -0
  284. data/app/components/shadcn/toast_title_component.rb +12 -0
  285. data/app/components/shadcn/toast_viewport_component.rb +12 -0
  286. data/app/components/shadcn/toggle_component.rb +77 -0
  287. data/app/components/shadcn/toggle_group_component.rb +96 -0
  288. data/app/components/shadcn/toggle_group_item_component.rb +62 -0
  289. data/app/components/shadcn/tooltip_component.rb +89 -0
  290. data/app/components/shadcn/typography_component.rb +112 -0
  291. data/babel.config.cjs +5 -0
  292. data/bin/console +11 -0
  293. data/bin/setup +8 -0
  294. data/config/importmap.rb +5 -0
  295. data/fly.toml +26 -0
  296. data/jest.config.js +19 -0
  297. data/jest.setup.js +8 -0
  298. data/lib/generators/shadcn/component/component_generator.rb +188 -0
  299. data/lib/generators/shadcn/install/install_generator.rb +140 -0
  300. data/lib/generators/shadcn/install/templates/initializer.rb.tt +35 -0
  301. data/lib/generators/shadcn/install/templates/shadcn.yml.tt +35 -0
  302. data/lib/generators/shadcn/theme/theme_generator.rb +128 -0
  303. data/lib/shadcn/rails/class_merger.rb +228 -0
  304. data/lib/shadcn/rails/configuration.rb +341 -0
  305. data/lib/shadcn/rails/engine.rb +59 -0
  306. data/lib/shadcn/rails/helpers/class_name_helper.rb +35 -0
  307. data/lib/shadcn/rails/helpers/component_helper.rb +60 -0
  308. data/lib/shadcn/rails/helpers/pagination_helper.rb +187 -0
  309. data/lib/shadcn/rails/version.rb +7 -0
  310. data/lib/shadcn/rails.rb +179 -0
  311. data/package-lock.json +7415 -0
  312. data/package.json +68 -0
  313. data/rollup.config.js +29 -0
  314. data/sig/shadcn/rails.rbs +6 -0
  315. metadata +526 -0
@@ -0,0 +1,454 @@
1
+ import { Application } from "@hotwired/stimulus"
2
+ import CheckboxController from "../../app/assets/javascripts/shadcn/controllers/checkbox_controller.js"
3
+ import { setupController, cleanupController, click, nextFrame, keydown } from '../helpers/stimulus-test-helper.js'
4
+
5
+ describe("CheckboxController", () => {
6
+ let application
7
+ let element
8
+ let controller
9
+
10
+ afterEach(() => {
11
+ cleanupController(application)
12
+ })
13
+
14
+ describe("basic rendering and initialization", () => {
15
+ const basicHTML = `
16
+ <button data-controller="shadcn--checkbox"
17
+ data-shadcn--checkbox-checked-value="false"
18
+ data-shadcn--checkbox-name-value="agree"
19
+ type="button"
20
+ role="checkbox"
21
+ aria-checked="false"
22
+ data-action="click->shadcn--checkbox#toggle">
23
+ <svg style="opacity: 0;">✓</svg>
24
+ </button>
25
+ `
26
+
27
+ beforeEach(async () => {
28
+ const setup = await setupController(CheckboxController, basicHTML, 'shadcn--checkbox')
29
+ application = setup.application
30
+ element = setup.element
31
+ controller = setup.controller
32
+ })
33
+
34
+ test("initializes with unchecked state", () => {
35
+ expect(controller.checkedValue).toBe(false)
36
+ })
37
+
38
+ test("sets data-state unchecked on element", () => {
39
+ expect(element.dataset.state).toBe("unchecked")
40
+ })
41
+
42
+ test("sets aria-checked false on element", () => {
43
+ expect(element.getAttribute("aria-checked")).toBe("false")
44
+ })
45
+
46
+ test("hides checkmark icon initially", () => {
47
+ const svg = element.querySelector("svg")
48
+ expect(svg.style.opacity).toBe("0")
49
+ })
50
+
51
+ test("initializes with name value", () => {
52
+ expect(controller.nameValue).toBe("agree")
53
+ })
54
+ })
55
+
56
+ describe("toggle functionality", () => {
57
+ const toggleHTML = `
58
+ <div>
59
+ <button data-controller="shadcn--checkbox"
60
+ data-shadcn--checkbox-checked-value="false"
61
+ data-shadcn--checkbox-name-value="terms"
62
+ type="button"
63
+ role="checkbox"
64
+ data-action="click->shadcn--checkbox#toggle">
65
+ <svg style="opacity: 0;">✓</svg>
66
+ </button>
67
+ <input type="hidden" name="terms" value="0">
68
+ </div>
69
+ `
70
+
71
+ beforeEach(async () => {
72
+ const setup = await setupController(CheckboxController, toggleHTML, 'shadcn--checkbox')
73
+ application = setup.application
74
+ element = setup.element
75
+ controller = setup.controller
76
+ })
77
+
78
+ test("toggles from unchecked to checked", async () => {
79
+ controller.toggle()
80
+ await nextFrame()
81
+
82
+ expect(controller.checkedValue).toBe(true)
83
+ })
84
+
85
+ test("toggles from checked to unchecked", async () => {
86
+ controller.checkedValue = true
87
+ controller.toggle()
88
+ await nextFrame()
89
+
90
+ expect(controller.checkedValue).toBe(false)
91
+ })
92
+
93
+ test("updates data-state on toggle", async () => {
94
+ controller.toggle()
95
+ await nextFrame()
96
+
97
+ expect(element.dataset.state).toBe("checked")
98
+ })
99
+
100
+ test("updates aria-checked on toggle", async () => {
101
+ controller.toggle()
102
+ await nextFrame()
103
+
104
+ expect(element.getAttribute("aria-checked")).toBe("true")
105
+ })
106
+
107
+ test("shows checkmark icon when checked", async () => {
108
+ controller.toggle()
109
+ await nextFrame()
110
+
111
+ const svg = element.querySelector("svg")
112
+ expect(svg.style.opacity).toBe("1")
113
+ })
114
+
115
+ test("hides checkmark icon when unchecked", async () => {
116
+ controller.checkedValue = true
117
+ controller.updateState()
118
+ controller.toggle()
119
+ await nextFrame()
120
+
121
+ const svg = element.querySelector("svg")
122
+ expect(svg.style.opacity).toBe("0")
123
+ })
124
+
125
+ test("dispatches change event on toggle", async () => {
126
+ let eventDetail = null
127
+ element.addEventListener("shadcn--checkbox:change", (e) => {
128
+ eventDetail = e.detail
129
+ })
130
+
131
+ controller.toggle()
132
+ await nextFrame()
133
+
134
+ expect(eventDetail).not.toBeNull()
135
+ expect(eventDetail.checked).toBe(true)
136
+ })
137
+
138
+ test("dispatches change event with false when unchecking", async () => {
139
+ controller.checkedValue = true
140
+ let eventDetail = null
141
+ element.addEventListener("shadcn--checkbox:change", (e) => {
142
+ eventDetail = e.detail
143
+ })
144
+
145
+ controller.toggle()
146
+ await nextFrame()
147
+
148
+ expect(eventDetail.checked).toBe(false)
149
+ })
150
+ })
151
+
152
+ describe("hidden input synchronization", () => {
153
+ const inputSyncHTML = `
154
+ <div>
155
+ <button data-controller="shadcn--checkbox"
156
+ data-shadcn--checkbox-checked-value="false"
157
+ data-shadcn--checkbox-name-value="subscribe"
158
+ type="button"
159
+ data-action="click->shadcn--checkbox#toggle">
160
+ <svg style="opacity: 0;">✓</svg>
161
+ </button>
162
+ <input type="hidden" name="subscribe" value="0">
163
+ </div>
164
+ `
165
+
166
+ beforeEach(async () => {
167
+ const setup = await setupController(CheckboxController, inputSyncHTML, 'shadcn--checkbox')
168
+ application = setup.application
169
+ element = setup.element
170
+ controller = setup.controller
171
+ })
172
+
173
+ test("updates hidden input to 1 when checked", async () => {
174
+ controller.toggle()
175
+ await nextFrame()
176
+
177
+ const input = element.parentElement.querySelector('input[name="subscribe"]')
178
+ expect(input.value).toBe("1")
179
+ })
180
+
181
+ test("updates hidden input to 0 when unchecked", async () => {
182
+ controller.checkedValue = true
183
+ controller.updateHiddenInput()
184
+
185
+ controller.toggle()
186
+ await nextFrame()
187
+
188
+ const input = element.parentElement.querySelector('input[name="subscribe"]')
189
+ expect(input.value).toBe("0")
190
+ })
191
+ })
192
+
193
+ describe("initial checked state", () => {
194
+ const checkedHTML = `
195
+ <div>
196
+ <button data-controller="shadcn--checkbox"
197
+ data-shadcn--checkbox-checked-value="true"
198
+ data-shadcn--checkbox-name-value="remember"
199
+ type="button"
200
+ role="checkbox"
201
+ aria-checked="true"
202
+ data-action="click->shadcn--checkbox#toggle">
203
+ <svg style="opacity: 1;">✓</svg>
204
+ </button>
205
+ <input type="hidden" name="remember" value="1">
206
+ </div>
207
+ `
208
+
209
+ beforeEach(async () => {
210
+ const setup = await setupController(CheckboxController, checkedHTML, 'shadcn--checkbox')
211
+ application = setup.application
212
+ element = setup.element
213
+ controller = setup.controller
214
+ })
215
+
216
+ test("initializes with checked state", () => {
217
+ expect(controller.checkedValue).toBe(true)
218
+ })
219
+
220
+ test("sets data-state checked on init", () => {
221
+ expect(element.dataset.state).toBe("checked")
222
+ })
223
+
224
+ test("sets aria-checked true on init", () => {
225
+ expect(element.getAttribute("aria-checked")).toBe("true")
226
+ })
227
+
228
+ test("shows checkmark on init", () => {
229
+ const svg = element.querySelector("svg")
230
+ expect(svg.style.opacity).toBe("1")
231
+ })
232
+ })
233
+
234
+ describe("programmatic value change", () => {
235
+ const programmaticHTML = `
236
+ <button data-controller="shadcn--checkbox"
237
+ data-shadcn--checkbox-checked-value="false"
238
+ type="button">
239
+ <svg style="opacity: 0;">✓</svg>
240
+ </button>
241
+ `
242
+
243
+ beforeEach(async () => {
244
+ const setup = await setupController(CheckboxController, programmaticHTML, 'shadcn--checkbox')
245
+ application = setup.application
246
+ element = setup.element
247
+ controller = setup.controller
248
+ })
249
+
250
+ test("updates UI when checkedValue changes", async () => {
251
+ controller.checkedValue = true
252
+ await nextFrame()
253
+
254
+ expect(element.dataset.state).toBe("checked")
255
+ expect(element.getAttribute("aria-checked")).toBe("true")
256
+ })
257
+
258
+ test("checkedValueChanged callback updates state", async () => {
259
+ const updateStateSpy = jest.spyOn(controller, 'updateState')
260
+
261
+ controller.checkedValue = true
262
+ await nextFrame()
263
+
264
+ expect(updateStateSpy).toHaveBeenCalled()
265
+ })
266
+ })
267
+
268
+ describe("without name value", () => {
269
+ const noNameHTML = `
270
+ <div>
271
+ <button data-controller="shadcn--checkbox"
272
+ data-shadcn--checkbox-checked-value="false"
273
+ type="button"
274
+ data-action="click->shadcn--checkbox#toggle">
275
+ <svg style="opacity: 0;">✓</svg>
276
+ </button>
277
+ </div>
278
+ `
279
+
280
+ beforeEach(async () => {
281
+ const setup = await setupController(CheckboxController, noNameHTML, 'shadcn--checkbox')
282
+ application = setup.application
283
+ element = setup.element
284
+ controller = setup.controller
285
+ })
286
+
287
+ test("works without name value", async () => {
288
+ expect(() => {
289
+ controller.toggle()
290
+ }).not.toThrow()
291
+
292
+ expect(controller.checkedValue).toBe(true)
293
+ })
294
+
295
+ test("does not try to update non-existent hidden input", () => {
296
+ expect(() => {
297
+ controller.updateHiddenInput()
298
+ }).not.toThrow()
299
+ })
300
+ })
301
+
302
+ describe("without SVG icon", () => {
303
+ const noIconHTML = `
304
+ <button data-controller="shadcn--checkbox"
305
+ data-shadcn--checkbox-checked-value="false"
306
+ type="button"
307
+ data-action="click->shadcn--checkbox#toggle">
308
+ Check
309
+ </button>
310
+ `
311
+
312
+ beforeEach(async () => {
313
+ const setup = await setupController(CheckboxController, noIconHTML, 'shadcn--checkbox')
314
+ application = setup.application
315
+ element = setup.element
316
+ controller = setup.controller
317
+ })
318
+
319
+ test("works without SVG icon", async () => {
320
+ expect(() => {
321
+ controller.toggle()
322
+ }).not.toThrow()
323
+
324
+ expect(controller.checkedValue).toBe(true)
325
+ })
326
+
327
+ test("updateState does not throw without SVG", () => {
328
+ expect(() => {
329
+ controller.updateState()
330
+ }).not.toThrow()
331
+ })
332
+ })
333
+
334
+ describe("multiple toggles", () => {
335
+ const multipleHTML = `
336
+ <button data-controller="shadcn--checkbox"
337
+ data-shadcn--checkbox-checked-value="false"
338
+ type="button"
339
+ data-action="click->shadcn--checkbox#toggle">
340
+ <svg style="opacity: 0;">✓</svg>
341
+ </button>
342
+ `
343
+
344
+ beforeEach(async () => {
345
+ const setup = await setupController(CheckboxController, multipleHTML, 'shadcn--checkbox')
346
+ application = setup.application
347
+ element = setup.element
348
+ controller = setup.controller
349
+ })
350
+
351
+ test("handles multiple rapid toggles", async () => {
352
+ controller.toggle() // true
353
+ controller.toggle() // false
354
+ controller.toggle() // true
355
+ await nextFrame()
356
+
357
+ expect(controller.checkedValue).toBe(true)
358
+ })
359
+
360
+ test("emits change event for each toggle", async () => {
361
+ let changeCount = 0
362
+ element.addEventListener("shadcn--checkbox:change", () => {
363
+ changeCount++
364
+ })
365
+
366
+ controller.toggle()
367
+ controller.toggle()
368
+ controller.toggle()
369
+ await nextFrame()
370
+
371
+ expect(changeCount).toBe(3)
372
+ })
373
+
374
+ test("final state is correct after even number of toggles", async () => {
375
+ controller.toggle()
376
+ controller.toggle()
377
+ controller.toggle()
378
+ controller.toggle()
379
+ await nextFrame()
380
+
381
+ expect(controller.checkedValue).toBe(false)
382
+ })
383
+ })
384
+
385
+ describe("click handler", () => {
386
+ const clickHTML = `
387
+ <button data-controller="shadcn--checkbox"
388
+ data-shadcn--checkbox-checked-value="false"
389
+ type="button"
390
+ data-action="click->shadcn--checkbox#toggle">
391
+ <svg style="opacity: 0;">✓</svg>
392
+ </button>
393
+ `
394
+
395
+ beforeEach(async () => {
396
+ const setup = await setupController(CheckboxController, clickHTML, 'shadcn--checkbox')
397
+ application = setup.application
398
+ element = setup.element
399
+ controller = setup.controller
400
+ })
401
+
402
+ test("toggles on click", async () => {
403
+ click(element)
404
+ await nextFrame()
405
+
406
+ expect(controller.checkedValue).toBe(true)
407
+ })
408
+
409
+ test("toggles again on second click", async () => {
410
+ click(element)
411
+ await nextFrame()
412
+ click(element)
413
+ await nextFrame()
414
+
415
+ expect(controller.checkedValue).toBe(false)
416
+ })
417
+ })
418
+
419
+ describe("edge cases", () => {
420
+ const edgeCaseHTML = `
421
+ <div>
422
+ <button data-controller="shadcn--checkbox"
423
+ data-shadcn--checkbox-checked-value="false"
424
+ data-shadcn--checkbox-name-value="test"
425
+ type="button"
426
+ data-action="click->shadcn--checkbox#toggle">
427
+ <svg style="opacity: 0;">✓</svg>
428
+ </button>
429
+ <input type="hidden" name="other" value="0">
430
+ </div>
431
+ `
432
+
433
+ beforeEach(async () => {
434
+ const setup = await setupController(CheckboxController, edgeCaseHTML, 'shadcn--checkbox')
435
+ application = setup.application
436
+ element = setup.element
437
+ controller = setup.controller
438
+ })
439
+
440
+ test("does not update input with different name", async () => {
441
+ controller.toggle()
442
+ await nextFrame()
443
+
444
+ const otherInput = element.parentElement.querySelector('input[name="other"]')
445
+ expect(otherInput.value).toBe("0")
446
+ })
447
+
448
+ test("gracefully handles missing input with matching name", async () => {
449
+ expect(() => {
450
+ controller.toggle()
451
+ }).not.toThrow()
452
+ })
453
+ })
454
+ })