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,251 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ /**
4
+ * Navigation Menu Controller
5
+ * Handles navigation menu interactions with dropdown content areas
6
+ */
7
+ export default class extends Controller {
8
+ static targets = ["list", "item", "trigger", "content", "viewport"]
9
+ static values = {
10
+ openIndex: { type: Number, default: -1 },
11
+ delayDuration: { type: Number, default: 200 },
12
+ skipDelayDuration: { type: Number, default: 300 }
13
+ }
14
+
15
+ connect() {
16
+ this.isOpen = false
17
+ this.previousIndex = -1
18
+ this.openTimer = null
19
+ this.closeTimer = null
20
+ this.wasClickOpened = false
21
+
22
+ this.boundHandleClickOutside = this.handleClickOutside.bind(this)
23
+ this.boundHandleKeydown = this.handleKeydown.bind(this)
24
+ }
25
+
26
+ disconnect() {
27
+ this.closeAll()
28
+ this.clearTimers()
29
+ }
30
+
31
+ toggle(event) {
32
+ event?.preventDefault()
33
+ const trigger = event.currentTarget
34
+ const item = trigger.closest("[data-shadcn--navigation-menu-target='item']")
35
+ const index = this.itemTargets.indexOf(item)
36
+
37
+ this.clearTimers()
38
+
39
+ if (this.openIndexValue === index) {
40
+ this.closeAll()
41
+ } else {
42
+ this.wasClickOpened = true
43
+ this.openItem(index)
44
+ }
45
+ }
46
+
47
+ hoverOpen(event) {
48
+ // If opened by click, require click to close
49
+ if (this.wasClickOpened && this.isOpen) return
50
+
51
+ const trigger = event.currentTarget
52
+ const item = trigger.closest("[data-shadcn--navigation-menu-target='item']")
53
+ const index = this.itemTargets.indexOf(item)
54
+
55
+ this.clearTimers()
56
+
57
+ if (this.isOpen) {
58
+ // Already open, switch immediately
59
+ if (this.openIndexValue !== index) {
60
+ this.openItem(index)
61
+ }
62
+ } else {
63
+ // Not open, delay before opening
64
+ this.openTimer = setTimeout(() => {
65
+ this.openItem(index)
66
+ }, this.delayDurationValue)
67
+ }
68
+ }
69
+
70
+ hoverClose(event) {
71
+ // If opened by click, require click to close
72
+ if (this.wasClickOpened) return
73
+
74
+ this.clearTimers()
75
+
76
+ this.closeTimer = setTimeout(() => {
77
+ this.closeAll()
78
+ }, this.skipDelayDurationValue)
79
+ }
80
+
81
+ contentHover() {
82
+ // Cancel close timer when hovering content
83
+ this.clearTimers()
84
+ }
85
+
86
+ openItem(index) {
87
+ if (index < 0 || index >= this.itemTargets.length) return
88
+
89
+ // Close previous if different
90
+ if (this.openIndexValue !== -1 && this.openIndexValue !== index) {
91
+ this.closeItem(this.openIndexValue)
92
+ }
93
+
94
+ const item = this.itemTargets[index]
95
+ const trigger = item.querySelector("[data-shadcn--navigation-menu-target='trigger']")
96
+ const content = item.querySelector("[data-shadcn--navigation-menu-target='content']")
97
+
98
+ if (!trigger || !content) return
99
+
100
+ this.previousIndex = this.openIndexValue
101
+ this.openIndexValue = index
102
+
103
+ // Update trigger state
104
+ trigger.setAttribute("aria-expanded", "true")
105
+ trigger.dataset.state = "open"
106
+
107
+ // Show content
108
+ content.hidden = false
109
+ content.dataset.state = "open"
110
+
111
+ // Set motion direction for animation
112
+ if (this.previousIndex !== -1 && this.previousIndex !== index) {
113
+ content.dataset.motion = this.previousIndex < index ? "from-end" : "from-start"
114
+ } else {
115
+ content.dataset.motion = "from-start"
116
+ }
117
+
118
+ // Update viewport
119
+ if (this.hasViewportTarget) {
120
+ this.viewportTarget.hidden = false
121
+ this.viewportTarget.dataset.state = "open"
122
+ this.viewportTarget.innerHTML = content.innerHTML
123
+ this.positionViewport(item)
124
+ }
125
+
126
+ this.isOpen = true
127
+
128
+ // Add event listeners
129
+ document.addEventListener("click", this.boundHandleClickOutside)
130
+ document.addEventListener("keydown", this.boundHandleKeydown)
131
+ }
132
+
133
+ closeItem(index) {
134
+ if (index < 0 || index >= this.itemTargets.length) return
135
+
136
+ const item = this.itemTargets[index]
137
+ const trigger = item.querySelector("[data-shadcn--navigation-menu-target='trigger']")
138
+ const content = item.querySelector("[data-shadcn--navigation-menu-target='content']")
139
+
140
+ if (trigger) {
141
+ trigger.setAttribute("aria-expanded", "false")
142
+ trigger.dataset.state = "closed"
143
+ }
144
+
145
+ if (content) {
146
+ content.dataset.state = "closed"
147
+ content.dataset.motion = this.previousIndex < index ? "to-end" : "to-start"
148
+ setTimeout(() => {
149
+ if (content.dataset.state === "closed") {
150
+ content.hidden = true
151
+ }
152
+ }, 150)
153
+ }
154
+ }
155
+
156
+ closeAll() {
157
+ this.triggerTargets.forEach(trigger => {
158
+ trigger.setAttribute("aria-expanded", "false")
159
+ trigger.dataset.state = "closed"
160
+ })
161
+
162
+ this.contentTargets.forEach(content => {
163
+ content.dataset.state = "closed"
164
+ setTimeout(() => {
165
+ if (content.dataset.state === "closed") {
166
+ content.hidden = true
167
+ }
168
+ }, 150)
169
+ })
170
+
171
+ if (this.hasViewportTarget) {
172
+ this.viewportTarget.dataset.state = "closed"
173
+ setTimeout(() => {
174
+ if (this.viewportTarget.dataset.state === "closed") {
175
+ this.viewportTarget.hidden = true
176
+ this.viewportTarget.innerHTML = ""
177
+ }
178
+ }, 150)
179
+ }
180
+
181
+ this.openIndexValue = -1
182
+ this.previousIndex = -1
183
+ this.isOpen = false
184
+ this.wasClickOpened = false
185
+
186
+ document.removeEventListener("click", this.boundHandleClickOutside)
187
+ document.removeEventListener("keydown", this.boundHandleKeydown)
188
+ }
189
+
190
+ handleClickOutside(event) {
191
+ if (!this.element.contains(event.target)) {
192
+ this.closeAll()
193
+ }
194
+ }
195
+
196
+ handleKeydown(event) {
197
+ switch (event.key) {
198
+ case "Escape":
199
+ this.closeAll()
200
+ if (this.openIndexValue >= 0) {
201
+ this.triggerTargets[this.openIndexValue]?.focus()
202
+ }
203
+ break
204
+ case "ArrowRight":
205
+ event.preventDefault()
206
+ this.navigateToNextItem()
207
+ break
208
+ case "ArrowLeft":
209
+ event.preventDefault()
210
+ this.navigateToPreviousItem()
211
+ break
212
+ }
213
+ }
214
+
215
+ navigateToNextItem() {
216
+ const nextIndex = (this.openIndexValue + 1) % this.itemTargets.length
217
+ this.openItem(nextIndex)
218
+ const trigger = this.itemTargets[nextIndex].querySelector("[data-shadcn--navigation-menu-target='trigger']")
219
+ trigger?.focus()
220
+ }
221
+
222
+ navigateToPreviousItem() {
223
+ const prevIndex = this.openIndexValue <= 0 ? this.itemTargets.length - 1 : this.openIndexValue - 1
224
+ this.openItem(prevIndex)
225
+ const trigger = this.itemTargets[prevIndex].querySelector("[data-shadcn--navigation-menu-target='trigger']")
226
+ trigger?.focus()
227
+ }
228
+
229
+ clearTimers() {
230
+ if (this.openTimer) {
231
+ clearTimeout(this.openTimer)
232
+ this.openTimer = null
233
+ }
234
+ if (this.closeTimer) {
235
+ clearTimeout(this.closeTimer)
236
+ this.closeTimer = null
237
+ }
238
+ }
239
+
240
+ positionViewport(item) {
241
+ if (!this.hasViewportTarget) return
242
+
243
+ const content = item.querySelector("[data-shadcn--navigation-menu-target='content']")
244
+ if (content) {
245
+ // Set CSS custom properties for width/height based on content
246
+ const rect = content.getBoundingClientRect()
247
+ this.viewportTarget.style.setProperty("--radix-navigation-menu-viewport-width", `${rect.width}px`)
248
+ this.viewportTarget.style.setProperty("--radix-navigation-menu-viewport-height", `${rect.height}px`)
249
+ }
250
+ }
251
+ }
@@ -0,0 +1,56 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+
3
+ /**
4
+ * Popover controller for rich content overlays
5
+ */
6
+ export default class PopoverController extends Controller {
7
+ static targets: ["trigger", "content"];
8
+ static values: {
9
+ open: { type: "Boolean"; default: false };
10
+ side: { type: "String"; default: "bottom" };
11
+ align: { type: "String"; default: "center" };
12
+ modal: { type: "Boolean"; default: false };
13
+ };
14
+
15
+ /** Popover trigger target */
16
+ readonly triggerTarget: HTMLElement;
17
+ readonly hasTriggerTarget: boolean;
18
+
19
+ /** Popover content target */
20
+ readonly contentTarget: HTMLElement;
21
+ readonly hasContentTarget: boolean;
22
+
23
+ /** Whether the popover is open */
24
+ openValue: boolean;
25
+ readonly hasOpenValue: boolean;
26
+
27
+ /** Side to display: "top", "bottom", "left", "right" */
28
+ sideValue: "top" | "bottom" | "left" | "right";
29
+ readonly hasSideValue: boolean;
30
+
31
+ /** Horizontal alignment: "start", "center", "end" */
32
+ alignValue: "start" | "center" | "end";
33
+ readonly hasAlignValue: boolean;
34
+
35
+ /** Whether the popover is modal */
36
+ modalValue: boolean;
37
+ readonly hasModalValue: boolean;
38
+
39
+ /** Toggle popover open/closed */
40
+ toggle(event?: Event): void;
41
+
42
+ /** Show the popover */
43
+ show(): void;
44
+
45
+ /** Hide the popover */
46
+ hide(): void;
47
+
48
+ /** Close the popover (alias for hide) */
49
+ close(): void;
50
+
51
+ /** Handle clicks outside the popover */
52
+ handleClickOutside(event: MouseEvent): void;
53
+
54
+ /** Position the popover content */
55
+ positionContent(): void;
56
+ }
@@ -0,0 +1,141 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ /**
4
+ * Popover controller for rich content overlays
5
+ */
6
+ export default class extends Controller {
7
+ static targets = ["trigger", "content"]
8
+ static values = {
9
+ open: { type: Boolean, default: false },
10
+ side: { type: String, default: "bottom" },
11
+ align: { type: String, default: "center" },
12
+ modal: { type: Boolean, default: false }
13
+ }
14
+
15
+ connect() {
16
+ this.boundHandleClickOutside = this.handleClickOutside.bind(this)
17
+
18
+ if (this.openValue) {
19
+ this.show()
20
+ }
21
+ }
22
+
23
+ disconnect() {
24
+ this.hide()
25
+ }
26
+
27
+ toggle(event) {
28
+ event?.preventDefault()
29
+ if (this.openValue) {
30
+ this.hide()
31
+ } else {
32
+ this.show()
33
+ }
34
+ }
35
+
36
+ show() {
37
+ if (this.openValue) return
38
+
39
+ this.openValue = true
40
+
41
+ if (this.hasContentTarget) {
42
+ this.contentTarget.hidden = false
43
+ this.contentTarget.dataset.state = "open"
44
+ this.contentTarget.dataset.side = this.sideValue
45
+ this.positionContent()
46
+ }
47
+
48
+ document.addEventListener("click", this.boundHandleClickOutside)
49
+
50
+ if (this.modalValue) {
51
+ document.body.style.pointerEvents = "none"
52
+ this.contentTarget.style.pointerEvents = "auto"
53
+ }
54
+
55
+ this.dispatch("opened")
56
+ }
57
+
58
+ hide() {
59
+ if (!this.openValue) return
60
+
61
+ this.openValue = false
62
+
63
+ if (this.hasContentTarget) {
64
+ this.contentTarget.dataset.state = "closed"
65
+ setTimeout(() => {
66
+ if (!this.openValue) {
67
+ this.contentTarget.hidden = true
68
+ }
69
+ }, 150)
70
+ }
71
+
72
+ document.removeEventListener("click", this.boundHandleClickOutside)
73
+
74
+ if (this.modalValue) {
75
+ document.body.style.pointerEvents = ""
76
+ }
77
+
78
+ this.dispatch("closed")
79
+ }
80
+
81
+ close() {
82
+ this.hide()
83
+ }
84
+
85
+ handleClickOutside(event) {
86
+ if (!this.element.contains(event.target)) {
87
+ this.hide()
88
+ }
89
+ }
90
+
91
+ positionContent() {
92
+ if (!this.hasContentTarget || !this.hasTriggerTarget) return
93
+
94
+ const trigger = this.triggerTarget.getBoundingClientRect()
95
+ const content = this.contentTarget
96
+
97
+ content.style.position = "absolute"
98
+
99
+ const gap = 8
100
+
101
+ switch (this.sideValue) {
102
+ case "top":
103
+ content.style.bottom = "100%"
104
+ content.style.top = "auto"
105
+ content.style.marginBottom = `${gap}px`
106
+ break
107
+ case "bottom":
108
+ content.style.top = "100%"
109
+ content.style.bottom = "auto"
110
+ content.style.marginTop = `${gap}px`
111
+ break
112
+ case "left":
113
+ content.style.right = "100%"
114
+ content.style.left = "auto"
115
+ content.style.marginRight = `${gap}px`
116
+ break
117
+ case "right":
118
+ content.style.left = "100%"
119
+ content.style.right = "auto"
120
+ content.style.marginLeft = `${gap}px`
121
+ break
122
+ }
123
+
124
+ switch (this.alignValue) {
125
+ case "start":
126
+ content.style.left = "0"
127
+ content.style.right = "auto"
128
+ break
129
+ case "center":
130
+ if (this.sideValue === "top" || this.sideValue === "bottom") {
131
+ content.style.left = "50%"
132
+ content.style.transform = "translateX(-50%)"
133
+ }
134
+ break
135
+ case "end":
136
+ content.style.right = "0"
137
+ content.style.left = "auto"
138
+ break
139
+ }
140
+ }
141
+ }
@@ -0,0 +1,47 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+
3
+ /**
4
+ * Radio Group Controller
5
+ * Handles radio group selection with keyboard navigation
6
+ */
7
+ export default class RadioGroupController extends Controller {
8
+ static targets: ["item", "indicator"];
9
+ static values: {
10
+ name: "String";
11
+ value: "String";
12
+ };
13
+
14
+ /** Radio item targets */
15
+ readonly itemTargets: HTMLElement[];
16
+ readonly hasItemTarget: boolean;
17
+
18
+ /** Visual indicator targets */
19
+ readonly indicatorTargets: HTMLElement[];
20
+ readonly hasIndicatorTarget: boolean;
21
+
22
+ /** Input name for form submission */
23
+ nameValue: string;
24
+ readonly hasNameValue: boolean;
25
+
26
+ /** Currently selected value */
27
+ valueValue: string;
28
+ readonly hasValueValue: boolean;
29
+
30
+ /** Select a radio item */
31
+ select(event: Event): void;
32
+
33
+ /** Handle keyboard navigation */
34
+ handleKeydown(event: KeyboardEvent): void;
35
+
36
+ /** Update selection state visuals */
37
+ updateSelection(): void;
38
+
39
+ /** Dispatch change event */
40
+ dispatchChange(value: string): void;
41
+
42
+ /** Get enabled (non-disabled) items */
43
+ readonly enabledItems: HTMLElement[];
44
+
45
+ /** Called when valueValue changes */
46
+ valueValueChanged(): void;
47
+ }
@@ -0,0 +1,108 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ /**
4
+ * Radio Group Controller
5
+ *
6
+ * Handles radio group selection with keyboard navigation
7
+ *
8
+ * Targets:
9
+ * - item: Individual radio buttons
10
+ * - indicator: Visual indicator element
11
+ *
12
+ * Values:
13
+ * - name: Input name for form submission
14
+ * - value: Currently selected value
15
+ */
16
+ export default class extends Controller {
17
+ static targets = ["item", "indicator"]
18
+ static values = {
19
+ name: String,
20
+ value: String
21
+ }
22
+
23
+ connect() {
24
+ this.updateSelection()
25
+ }
26
+
27
+ select(event) {
28
+ const item = event.currentTarget
29
+ if (item.disabled) return
30
+
31
+ const value = item.dataset.value
32
+ this.valueValue = value
33
+ this.updateSelection()
34
+ this.dispatchChange(value)
35
+ }
36
+
37
+ handleKeydown(event) {
38
+ const items = this.enabledItems
39
+ const currentIndex = items.indexOf(event.currentTarget)
40
+ let newIndex = currentIndex
41
+
42
+ switch (event.key) {
43
+ case "ArrowDown":
44
+ case "ArrowRight":
45
+ event.preventDefault()
46
+ newIndex = (currentIndex + 1) % items.length
47
+ break
48
+ case "ArrowUp":
49
+ case "ArrowLeft":
50
+ event.preventDefault()
51
+ newIndex = (currentIndex - 1 + items.length) % items.length
52
+ break
53
+ case " ":
54
+ case "Enter":
55
+ event.preventDefault()
56
+ this.select(event)
57
+ return
58
+ default:
59
+ return
60
+ }
61
+
62
+ const newItem = items[newIndex]
63
+ newItem.focus()
64
+ // Auto-select on arrow navigation (standard radio behavior)
65
+ this.valueValue = newItem.dataset.value
66
+ this.updateSelection()
67
+ this.dispatchChange(newItem.dataset.value)
68
+ }
69
+
70
+ updateSelection() {
71
+ this.itemTargets.forEach(item => {
72
+ const isSelected = item.dataset.value === this.valueValue
73
+ item.setAttribute("aria-checked", isSelected.toString())
74
+ item.dataset.state = isSelected ? "checked" : "unchecked"
75
+ item.tabIndex = isSelected ? 0 : -1
76
+
77
+ // Update indicator visibility
78
+ const indicator = item.querySelector("[data-shadcn--radio-group-target='indicator']")
79
+ if (indicator) {
80
+ indicator.classList.toggle("opacity-0", !isSelected)
81
+ }
82
+ })
83
+
84
+ // Ensure at least one item is focusable if nothing selected
85
+ if (!this.valueValue && this.itemTargets.length > 0) {
86
+ this.enabledItems[0]?.setAttribute("tabindex", "0")
87
+ }
88
+ }
89
+
90
+ dispatchChange(value) {
91
+ this.dispatch("change", {
92
+ detail: { value, name: this.nameValue }
93
+ })
94
+
95
+ // Also dispatch a native input event for form compatibility
96
+ const event = new Event("input", { bubbles: true })
97
+ this.element.dispatchEvent(event)
98
+ }
99
+
100
+ get enabledItems() {
101
+ return this.itemTargets.filter(item => !item.disabled)
102
+ }
103
+
104
+ // Allow programmatic value setting
105
+ valueValueChanged() {
106
+ this.updateSelection()
107
+ }
108
+ }