hotwirebits 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 (843) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG.md +46 -0
  3. data/LICENSE +21 -0
  4. data/README.md +156 -0
  5. data/app/assets/stylesheets/hotwirebits/theme.css +131 -0
  6. data/app/assets/stylesheets/hotwirebits/themes/corporate.css +29 -0
  7. data/app/assets/stylesheets/hotwirebits/themes/midnight.css +29 -0
  8. data/app/components/hotwirebits/accordion_component.html.erb +23 -0
  9. data/app/components/hotwirebits/accordion_component.rb +34 -0
  10. data/app/components/hotwirebits/activity_feed_component.html.erb +40 -0
  11. data/app/components/hotwirebits/activity_feed_component.rb +11 -0
  12. data/app/components/hotwirebits/affix_component.html.erb +3 -0
  13. data/app/components/hotwirebits/affix_component.rb +25 -0
  14. data/app/components/hotwirebits/ai_chat_component.html.erb +32 -0
  15. data/app/components/hotwirebits/ai_chat_component.rb +12 -0
  16. data/app/components/hotwirebits/alert_component.html.erb +32 -0
  17. data/app/components/hotwirebits/alert_component.rb +39 -0
  18. data/app/components/hotwirebits/alert_dialog_component.html.erb +30 -0
  19. data/app/components/hotwirebits/alert_dialog_component.rb +40 -0
  20. data/app/components/hotwirebits/anchor_component.html.erb +9 -0
  21. data/app/components/hotwirebits/anchor_component.rb +12 -0
  22. data/app/components/hotwirebits/angle_slider_component.html.erb +9 -0
  23. data/app/components/hotwirebits/angle_slider_component.rb +19 -0
  24. data/app/components/hotwirebits/animated_number_component.html.erb +8 -0
  25. data/app/components/hotwirebits/animated_number_component.rb +28 -0
  26. data/app/components/hotwirebits/app_shell_component.html.erb +22 -0
  27. data/app/components/hotwirebits/app_shell_component.rb +15 -0
  28. data/app/components/hotwirebits/aspect_ratio_component.html.erb +3 -0
  29. data/app/components/hotwirebits/aspect_ratio_component.rb +27 -0
  30. data/app/components/hotwirebits/auth_block_component.html.erb +63 -0
  31. data/app/components/hotwirebits/auth_block_component.rb +35 -0
  32. data/app/components/hotwirebits/autocomplete_component.html.erb +5 -0
  33. data/app/components/hotwirebits/autocomplete_component.rb +45 -0
  34. data/app/components/hotwirebits/avatar_component.html.erb +18 -0
  35. data/app/components/hotwirebits/avatar_component.rb +67 -0
  36. data/app/components/hotwirebits/background_image_component.html.erb +9 -0
  37. data/app/components/hotwirebits/background_image_component.rb +17 -0
  38. data/app/components/hotwirebits/badge_component.html.erb +13 -0
  39. data/app/components/hotwirebits/badge_component.rb +55 -0
  40. data/app/components/hotwirebits/banner_component.html.erb +31 -0
  41. data/app/components/hotwirebits/banner_component.rb +36 -0
  42. data/app/components/hotwirebits/base.rb +13 -0
  43. data/app/components/hotwirebits/bento_grid_component.html.erb +7 -0
  44. data/app/components/hotwirebits/bento_grid_component.rb +36 -0
  45. data/app/components/hotwirebits/blog_section_component.html.erb +35 -0
  46. data/app/components/hotwirebits/blog_section_component.rb +20 -0
  47. data/app/components/hotwirebits/bottom_navigation_component.html.erb +21 -0
  48. data/app/components/hotwirebits/bottom_navigation_component.rb +22 -0
  49. data/app/components/hotwirebits/box_component.html.erb +3 -0
  50. data/app/components/hotwirebits/box_component.rb +16 -0
  51. data/app/components/hotwirebits/breadcrumb_advanced_component.html.erb +53 -0
  52. data/app/components/hotwirebits/breadcrumb_advanced_component.rb +32 -0
  53. data/app/components/hotwirebits/breadcrumb_component.html.erb +18 -0
  54. data/app/components/hotwirebits/breadcrumb_component.rb +11 -0
  55. data/app/components/hotwirebits/browser_mockup_component.html.erb +22 -0
  56. data/app/components/hotwirebits/browser_mockup_component.rb +11 -0
  57. data/app/components/hotwirebits/burger_component.html.erb +20 -0
  58. data/app/components/hotwirebits/burger_component.rb +29 -0
  59. data/app/components/hotwirebits/button_component.html.erb +21 -0
  60. data/app/components/hotwirebits/button_component.rb +84 -0
  61. data/app/components/hotwirebits/calendar_component.html.erb +70 -0
  62. data/app/components/hotwirebits/calendar_component.rb +49 -0
  63. data/app/components/hotwirebits/card_component.html.erb +24 -0
  64. data/app/components/hotwirebits/card_component.rb +21 -0
  65. data/app/components/hotwirebits/card_heading_component.html.erb +13 -0
  66. data/app/components/hotwirebits/card_heading_component.rb +13 -0
  67. data/app/components/hotwirebits/carousel_component.html.erb +50 -0
  68. data/app/components/hotwirebits/carousel_component.rb +27 -0
  69. data/app/components/hotwirebits/center_component.html.erb +3 -0
  70. data/app/components/hotwirebits/center_component.rb +22 -0
  71. data/app/components/hotwirebits/chart_component.html.erb +68 -0
  72. data/app/components/hotwirebits/chart_component.rb +67 -0
  73. data/app/components/hotwirebits/chat_bubble_component.html.erb +27 -0
  74. data/app/components/hotwirebits/chat_bubble_component.rb +35 -0
  75. data/app/components/hotwirebits/checkbox_component.html.erb +9 -0
  76. data/app/components/hotwirebits/checkbox_component.rb +70 -0
  77. data/app/components/hotwirebits/checkout_form_component.html.erb +40 -0
  78. data/app/components/hotwirebits/checkout_form_component.rb +21 -0
  79. data/app/components/hotwirebits/chip_component.html.erb +13 -0
  80. data/app/components/hotwirebits/chip_component.rb +51 -0
  81. data/app/components/hotwirebits/clipboard_component.html.erb +26 -0
  82. data/app/components/hotwirebits/clipboard_component.rb +19 -0
  83. data/app/components/hotwirebits/close_button_component.html.erb +3 -0
  84. data/app/components/hotwirebits/close_button_component.rb +40 -0
  85. data/app/components/hotwirebits/code_block_component.html.erb +17 -0
  86. data/app/components/hotwirebits/code_block_component.rb +19 -0
  87. data/app/components/hotwirebits/code_highlight_component.html.erb +6 -0
  88. data/app/components/hotwirebits/code_highlight_component.rb +32 -0
  89. data/app/components/hotwirebits/code_mockup_component.html.erb +13 -0
  90. data/app/components/hotwirebits/code_mockup_component.rb +19 -0
  91. data/app/components/hotwirebits/collapsible_component.html.erb +17 -0
  92. data/app/components/hotwirebits/collapsible_component.rb +11 -0
  93. data/app/components/hotwirebits/color_input_component.html.erb +19 -0
  94. data/app/components/hotwirebits/color_input_component.rb +14 -0
  95. data/app/components/hotwirebits/color_picker_component.html.erb +6 -0
  96. data/app/components/hotwirebits/color_picker_component.rb +37 -0
  97. data/app/components/hotwirebits/color_swatch_component.html.erb +1 -0
  98. data/app/components/hotwirebits/color_swatch_component.rb +18 -0
  99. data/app/components/hotwirebits/combobox_component.html.erb +15 -0
  100. data/app/components/hotwirebits/combobox_component.rb +56 -0
  101. data/app/components/hotwirebits/command_bar_component.html.erb +33 -0
  102. data/app/components/hotwirebits/command_bar_component.rb +11 -0
  103. data/app/components/hotwirebits/compare_component.html.erb +21 -0
  104. data/app/components/hotwirebits/compare_component.rb +17 -0
  105. data/app/components/hotwirebits/confirm_dialog_component.html.erb +17 -0
  106. data/app/components/hotwirebits/confirm_dialog_component.rb +23 -0
  107. data/app/components/hotwirebits/confirm_popup_component.html.erb +16 -0
  108. data/app/components/hotwirebits/confirm_popup_component.rb +31 -0
  109. data/app/components/hotwirebits/contact_section_component.html.erb +53 -0
  110. data/app/components/hotwirebits/contact_section_component.rb +17 -0
  111. data/app/components/hotwirebits/container_component.html.erb +3 -0
  112. data/app/components/hotwirebits/container_component.rb +29 -0
  113. data/app/components/hotwirebits/context_menu_component.html.erb +27 -0
  114. data/app/components/hotwirebits/context_menu_component.rb +14 -0
  115. data/app/components/hotwirebits/cookie_consent_component.html.erb +23 -0
  116. data/app/components/hotwirebits/cookie_consent_component.rb +28 -0
  117. data/app/components/hotwirebits/cookie_settings_component.html.erb +48 -0
  118. data/app/components/hotwirebits/cookie_settings_component.rb +10 -0
  119. data/app/components/hotwirebits/countdown_component.html.erb +8 -0
  120. data/app/components/hotwirebits/countdown_component.rb +14 -0
  121. data/app/components/hotwirebits/cta_section_component.html.erb +33 -0
  122. data/app/components/hotwirebits/cta_section_component.rb +23 -0
  123. data/app/components/hotwirebits/dashboard_card_component.html.erb +32 -0
  124. data/app/components/hotwirebits/dashboard_card_component.rb +25 -0
  125. data/app/components/hotwirebits/data_table_component.html.erb +53 -0
  126. data/app/components/hotwirebits/data_table_component.rb +40 -0
  127. data/app/components/hotwirebits/data_view_component.html.erb +21 -0
  128. data/app/components/hotwirebits/data_view_component.rb +76 -0
  129. data/app/components/hotwirebits/date_picker_component.html.erb +8 -0
  130. data/app/components/hotwirebits/date_picker_component.rb +61 -0
  131. data/app/components/hotwirebits/date_range_picker_component.html.erb +28 -0
  132. data/app/components/hotwirebits/date_range_picker_component.rb +14 -0
  133. data/app/components/hotwirebits/device_frame_component.html.erb +40 -0
  134. data/app/components/hotwirebits/device_frame_component.rb +36 -0
  135. data/app/components/hotwirebits/device_mockup_component.html.erb +51 -0
  136. data/app/components/hotwirebits/device_mockup_component.rb +37 -0
  137. data/app/components/hotwirebits/dialog_component.html.erb +41 -0
  138. data/app/components/hotwirebits/dialog_component.rb +49 -0
  139. data/app/components/hotwirebits/diff_component.html.erb +22 -0
  140. data/app/components/hotwirebits/diff_component.rb +21 -0
  141. data/app/components/hotwirebits/diff_view_component.html.erb +22 -0
  142. data/app/components/hotwirebits/diff_view_component.rb +39 -0
  143. data/app/components/hotwirebits/digital_signature_component.html.erb +21 -0
  144. data/app/components/hotwirebits/digital_signature_component.rb +14 -0
  145. data/app/components/hotwirebits/divider_component.html.erb +11 -0
  146. data/app/components/hotwirebits/divider_component.rb +29 -0
  147. data/app/components/hotwirebits/dock_menu_component.html.erb +21 -0
  148. data/app/components/hotwirebits/dock_menu_component.rb +10 -0
  149. data/app/components/hotwirebits/drag_drop_list_component.html.erb +16 -0
  150. data/app/components/hotwirebits/drag_drop_list_component.rb +20 -0
  151. data/app/components/hotwirebits/drawer_component.html.erb +38 -0
  152. data/app/components/hotwirebits/drawer_component.rb +21 -0
  153. data/app/components/hotwirebits/dropdown_component.html.erb +27 -0
  154. data/app/components/hotwirebits/dropdown_component.rb +21 -0
  155. data/app/components/hotwirebits/dropzone_component.html.erb +16 -0
  156. data/app/components/hotwirebits/dropzone_component.rb +15 -0
  157. data/app/components/hotwirebits/empty_cart_component.html.erb +19 -0
  158. data/app/components/hotwirebits/empty_cart_component.rb +14 -0
  159. data/app/components/hotwirebits/empty_state_component.html.erb +20 -0
  160. data/app/components/hotwirebits/empty_state_component.rb +14 -0
  161. data/app/components/hotwirebits/error_page_component.html.erb +14 -0
  162. data/app/components/hotwirebits/error_page_component.rb +30 -0
  163. data/app/components/hotwirebits/faq_section_component.html.erb +26 -0
  164. data/app/components/hotwirebits/faq_section_component.rb +18 -0
  165. data/app/components/hotwirebits/feature_section_component.html.erb +27 -0
  166. data/app/components/hotwirebits/feature_section_component.rb +24 -0
  167. data/app/components/hotwirebits/feedback_component.html.erb +51 -0
  168. data/app/components/hotwirebits/feedback_component.rb +23 -0
  169. data/app/components/hotwirebits/fieldset_component.html.erb +11 -0
  170. data/app/components/hotwirebits/fieldset_component.rb +21 -0
  171. data/app/components/hotwirebits/file_browser_component.html.erb +35 -0
  172. data/app/components/hotwirebits/file_browser_component.rb +11 -0
  173. data/app/components/hotwirebits/file_input_component.html.erb +7 -0
  174. data/app/components/hotwirebits/file_input_component.rb +25 -0
  175. data/app/components/hotwirebits/file_upload_progress_component.html.erb +26 -0
  176. data/app/components/hotwirebits/file_upload_progress_component.rb +33 -0
  177. data/app/components/hotwirebits/flex_component.html.erb +3 -0
  178. data/app/components/hotwirebits/flex_component.rb +37 -0
  179. data/app/components/hotwirebits/float_label_component.html.erb +4 -0
  180. data/app/components/hotwirebits/float_label_component.rb +11 -0
  181. data/app/components/hotwirebits/fluid_component.html.erb +3 -0
  182. data/app/components/hotwirebits/fluid_component.rb +13 -0
  183. data/app/components/hotwirebits/flyout_menu_component.html.erb +35 -0
  184. data/app/components/hotwirebits/flyout_menu_component.rb +25 -0
  185. data/app/components/hotwirebits/footer_component.html.erb +43 -0
  186. data/app/components/hotwirebits/footer_component.rb +13 -0
  187. data/app/components/hotwirebits/form_fieldset_component.html.erb +9 -0
  188. data/app/components/hotwirebits/form_fieldset_component.rb +11 -0
  189. data/app/components/hotwirebits/form_group_component.html.erb +31 -0
  190. data/app/components/hotwirebits/form_group_component.rb +15 -0
  191. data/app/components/hotwirebits/gallery_component.html.erb +25 -0
  192. data/app/components/hotwirebits/gallery_component.rb +29 -0
  193. data/app/components/hotwirebits/grid_component.html.erb +3 -0
  194. data/app/components/hotwirebits/grid_component.rb +31 -0
  195. data/app/components/hotwirebits/group_component.html.erb +3 -0
  196. data/app/components/hotwirebits/group_component.rb +27 -0
  197. data/app/components/hotwirebits/header_component.html.erb +25 -0
  198. data/app/components/hotwirebits/header_component.rb +23 -0
  199. data/app/components/hotwirebits/header_section_component.html.erb +33 -0
  200. data/app/components/hotwirebits/header_section_component.rb +18 -0
  201. data/app/components/hotwirebits/hero_component.html.erb +35 -0
  202. data/app/components/hotwirebits/hero_component.rb +36 -0
  203. data/app/components/hotwirebits/highlight_component.html.erb +1 -0
  204. data/app/components/hotwirebits/highlight_component.rb +21 -0
  205. data/app/components/hotwirebits/hover3d_card_component.html.erb +10 -0
  206. data/app/components/hotwirebits/hover3d_card_component.rb +16 -0
  207. data/app/components/hotwirebits/hover_card_component.html.erb +19 -0
  208. data/app/components/hotwirebits/hover_card_component.rb +35 -0
  209. data/app/components/hotwirebits/hover_gallery_component.html.erb +23 -0
  210. data/app/components/hotwirebits/hover_gallery_component.rb +15 -0
  211. data/app/components/hotwirebits/icon_field_component.html.erb +9 -0
  212. data/app/components/hotwirebits/icon_field_component.rb +35 -0
  213. data/app/components/hotwirebits/ifta_label_component.html.erb +4 -0
  214. data/app/components/hotwirebits/ifta_label_component.rb +11 -0
  215. data/app/components/hotwirebits/image_compare_component.html.erb +21 -0
  216. data/app/components/hotwirebits/image_compare_component.rb +19 -0
  217. data/app/components/hotwirebits/image_component.html.erb +22 -0
  218. data/app/components/hotwirebits/image_component.rb +23 -0
  219. data/app/components/hotwirebits/image_zoom_component.html.erb +8 -0
  220. data/app/components/hotwirebits/image_zoom_component.rb +17 -0
  221. data/app/components/hotwirebits/incentive_component.html.erb +11 -0
  222. data/app/components/hotwirebits/incentive_component.rb +16 -0
  223. data/app/components/hotwirebits/indicator_badge_component.html.erb +8 -0
  224. data/app/components/hotwirebits/indicator_badge_component.rb +33 -0
  225. data/app/components/hotwirebits/indicator_component.html.erb +6 -0
  226. data/app/components/hotwirebits/indicator_component.rb +30 -0
  227. data/app/components/hotwirebits/inline_edit_component.html.erb +50 -0
  228. data/app/components/hotwirebits/inline_edit_component.rb +16 -0
  229. data/app/components/hotwirebits/inplace_component.html.erb +6 -0
  230. data/app/components/hotwirebits/inplace_component.rb +39 -0
  231. data/app/components/hotwirebits/input_component.html.erb +13 -0
  232. data/app/components/hotwirebits/input_component.rb +97 -0
  233. data/app/components/hotwirebits/input_group_component.html.erb +9 -0
  234. data/app/components/hotwirebits/input_group_component.rb +38 -0
  235. data/app/components/hotwirebits/input_number_component.html.erb +14 -0
  236. data/app/components/hotwirebits/input_number_component.rb +78 -0
  237. data/app/components/hotwirebits/input_otp_component.html.erb +6 -0
  238. data/app/components/hotwirebits/input_otp_component.rb +29 -0
  239. data/app/components/hotwirebits/input_tags_component.html.erb +12 -0
  240. data/app/components/hotwirebits/input_tags_component.rb +31 -0
  241. data/app/components/hotwirebits/join_component.html.erb +3 -0
  242. data/app/components/hotwirebits/join_component.rb +18 -0
  243. data/app/components/hotwirebits/json_input_component.html.erb +14 -0
  244. data/app/components/hotwirebits/json_input_component.rb +20 -0
  245. data/app/components/hotwirebits/kanban_board_component.html.erb +39 -0
  246. data/app/components/hotwirebits/kanban_board_component.rb +17 -0
  247. data/app/components/hotwirebits/kbd_component.html.erb +1 -0
  248. data/app/components/hotwirebits/kbd_component.rb +27 -0
  249. data/app/components/hotwirebits/knob_component.html.erb +16 -0
  250. data/app/components/hotwirebits/knob_component.rb +23 -0
  251. data/app/components/hotwirebits/label_component.html.erb +5 -0
  252. data/app/components/hotwirebits/label_component.rb +18 -0
  253. data/app/components/hotwirebits/lightbox_component.html.erb +30 -0
  254. data/app/components/hotwirebits/lightbox_component.rb +11 -0
  255. data/app/components/hotwirebits/like_button_component.html.erb +15 -0
  256. data/app/components/hotwirebits/like_button_component.rb +23 -0
  257. data/app/components/hotwirebits/list_component.html.erb +3 -0
  258. data/app/components/hotwirebits/list_component.rb +19 -0
  259. data/app/components/hotwirebits/listbox_component.html.erb +16 -0
  260. data/app/components/hotwirebits/listbox_component.rb +52 -0
  261. data/app/components/hotwirebits/loading_overlay_component.html.erb +12 -0
  262. data/app/components/hotwirebits/loading_overlay_component.rb +23 -0
  263. data/app/components/hotwirebits/logo_cloud_component.html.erb +20 -0
  264. data/app/components/hotwirebits/logo_cloud_component.rb +23 -0
  265. data/app/components/hotwirebits/logo_item_component.html.erb +7 -0
  266. data/app/components/hotwirebits/logo_item_component.rb +20 -0
  267. data/app/components/hotwirebits/mark_component.html.erb +1 -0
  268. data/app/components/hotwirebits/mark_component.rb +17 -0
  269. data/app/components/hotwirebits/marquee_component.html.erb +29 -0
  270. data/app/components/hotwirebits/marquee_component.rb +38 -0
  271. data/app/components/hotwirebits/mask_component.html.erb +3 -0
  272. data/app/components/hotwirebits/mask_component.rb +25 -0
  273. data/app/components/hotwirebits/mega_menu_component.html.erb +43 -0
  274. data/app/components/hotwirebits/mega_menu_component.rb +11 -0
  275. data/app/components/hotwirebits/menu_component.html.erb +32 -0
  276. data/app/components/hotwirebits/menu_component.rb +20 -0
  277. data/app/components/hotwirebits/menubar_component.html.erb +30 -0
  278. data/app/components/hotwirebits/menubar_component.rb +10 -0
  279. data/app/components/hotwirebits/meter_group_component.html.erb +16 -0
  280. data/app/components/hotwirebits/meter_group_component.rb +36 -0
  281. data/app/components/hotwirebits/month_picker_component.html.erb +25 -0
  282. data/app/components/hotwirebits/month_picker_component.rb +51 -0
  283. data/app/components/hotwirebits/month_picker_input_component.html.erb +20 -0
  284. data/app/components/hotwirebits/month_picker_input_component.rb +21 -0
  285. data/app/components/hotwirebits/multiselect_component.html.erb +18 -0
  286. data/app/components/hotwirebits/multiselect_component.rb +56 -0
  287. data/app/components/hotwirebits/native_select_component.html.erb +10 -0
  288. data/app/components/hotwirebits/native_select_component.rb +20 -0
  289. data/app/components/hotwirebits/nav_link_component.html.erb +9 -0
  290. data/app/components/hotwirebits/nav_link_component.rb +23 -0
  291. data/app/components/hotwirebits/navbar_component.html.erb +25 -0
  292. data/app/components/hotwirebits/navbar_component.rb +26 -0
  293. data/app/components/hotwirebits/navigation_menu_component.html.erb +43 -0
  294. data/app/components/hotwirebits/navigation_menu_component.rb +10 -0
  295. data/app/components/hotwirebits/newsletter_section_component.html.erb +18 -0
  296. data/app/components/hotwirebits/newsletter_section_component.rb +15 -0
  297. data/app/components/hotwirebits/notification_bell_component.html.erb +18 -0
  298. data/app/components/hotwirebits/notification_bell_component.rb +22 -0
  299. data/app/components/hotwirebits/notification_component.html.erb +28 -0
  300. data/app/components/hotwirebits/notification_component.rb +23 -0
  301. data/app/components/hotwirebits/number_formatter_component.html.erb +1 -0
  302. data/app/components/hotwirebits/number_formatter_component.rb +27 -0
  303. data/app/components/hotwirebits/number_input_component.html.erb +1 -0
  304. data/app/components/hotwirebits/number_input_component.rb +32 -0
  305. data/app/components/hotwirebits/order_summary_component.html.erb +37 -0
  306. data/app/components/hotwirebits/order_summary_component.rb +23 -0
  307. data/app/components/hotwirebits/org_chart_component.html.erb +34 -0
  308. data/app/components/hotwirebits/org_chart_component.rb +14 -0
  309. data/app/components/hotwirebits/page_heading_component.html.erb +28 -0
  310. data/app/components/hotwirebits/page_heading_component.rb +18 -0
  311. data/app/components/hotwirebits/pagination_component.html.erb +29 -0
  312. data/app/components/hotwirebits/pagination_component.rb +32 -0
  313. data/app/components/hotwirebits/paginator_component.html.erb +36 -0
  314. data/app/components/hotwirebits/paginator_component.rb +44 -0
  315. data/app/components/hotwirebits/panel_component.html.erb +14 -0
  316. data/app/components/hotwirebits/panel_component.rb +48 -0
  317. data/app/components/hotwirebits/password_input_component.html.erb +8 -0
  318. data/app/components/hotwirebits/password_input_component.rb +72 -0
  319. data/app/components/hotwirebits/password_strength_component.html.erb +18 -0
  320. data/app/components/hotwirebits/password_strength_component.rb +24 -0
  321. data/app/components/hotwirebits/password_toggle_component.html.erb +15 -0
  322. data/app/components/hotwirebits/password_toggle_component.rb +10 -0
  323. data/app/components/hotwirebits/permission_badge_component.html.erb +15 -0
  324. data/app/components/hotwirebits/permission_badge_component.rb +33 -0
  325. data/app/components/hotwirebits/phone_input_component.html.erb +1 -0
  326. data/app/components/hotwirebits/phone_input_component.rb +29 -0
  327. data/app/components/hotwirebits/phone_mockup_component.html.erb +12 -0
  328. data/app/components/hotwirebits/phone_mockup_component.rb +30 -0
  329. data/app/components/hotwirebits/popover_component.html.erb +20 -0
  330. data/app/components/hotwirebits/popover_component.rb +45 -0
  331. data/app/components/hotwirebits/portal_component.html.erb +3 -0
  332. data/app/components/hotwirebits/portal_component.rb +10 -0
  333. data/app/components/hotwirebits/pricing_section_component.html.erb +46 -0
  334. data/app/components/hotwirebits/pricing_section_component.rb +19 -0
  335. data/app/components/hotwirebits/product_list_component.html.erb +24 -0
  336. data/app/components/hotwirebits/product_list_component.rb +21 -0
  337. data/app/components/hotwirebits/product_overview_component.html.erb +59 -0
  338. data/app/components/hotwirebits/product_overview_component.rb +19 -0
  339. data/app/components/hotwirebits/product_quickview_component.html.erb +45 -0
  340. data/app/components/hotwirebits/product_quickview_component.rb +15 -0
  341. data/app/components/hotwirebits/progress_component.html.erb +10 -0
  342. data/app/components/hotwirebits/progress_component.rb +49 -0
  343. data/app/components/hotwirebits/progress_ring_component.html.erb +26 -0
  344. data/app/components/hotwirebits/progress_ring_component.rb +47 -0
  345. data/app/components/hotwirebits/promo_section_component.html.erb +31 -0
  346. data/app/components/hotwirebits/promo_section_component.rb +30 -0
  347. data/app/components/hotwirebits/qr_code_component.html.erb +19 -0
  348. data/app/components/hotwirebits/qr_code_component.rb +177 -0
  349. data/app/components/hotwirebits/radio_component.html.erb +6 -0
  350. data/app/components/hotwirebits/radio_component.rb +40 -0
  351. data/app/components/hotwirebits/range_slider_component.html.erb +9 -0
  352. data/app/components/hotwirebits/range_slider_component.rb +57 -0
  353. data/app/components/hotwirebits/rating_component.html.erb +11 -0
  354. data/app/components/hotwirebits/rating_component.rb +53 -0
  355. data/app/components/hotwirebits/resizable_component.html.erb +19 -0
  356. data/app/components/hotwirebits/resizable_component.rb +35 -0
  357. data/app/components/hotwirebits/review_component.html.erb +36 -0
  358. data/app/components/hotwirebits/review_component.rb +19 -0
  359. data/app/components/hotwirebits/rich_text_editor_component.html.erb +55 -0
  360. data/app/components/hotwirebits/rich_text_editor_component.rb +15 -0
  361. data/app/components/hotwirebits/scroll_area_component.html.erb +3 -0
  362. data/app/components/hotwirebits/scroll_area_component.rb +28 -0
  363. data/app/components/hotwirebits/search_component.html.erb +56 -0
  364. data/app/components/hotwirebits/search_component.rb +21 -0
  365. data/app/components/hotwirebits/search_input_component.html.erb +6 -0
  366. data/app/components/hotwirebits/search_input_component.rb +50 -0
  367. data/app/components/hotwirebits/section_heading_component.html.erb +6 -0
  368. data/app/components/hotwirebits/section_heading_component.rb +26 -0
  369. data/app/components/hotwirebits/select_component.html.erb +16 -0
  370. data/app/components/hotwirebits/select_component.rb +38 -0
  371. data/app/components/hotwirebits/semi_circle_progress_component.html.erb +9 -0
  372. data/app/components/hotwirebits/semi_circle_progress_component.rb +52 -0
  373. data/app/components/hotwirebits/separator_component.html.erb +9 -0
  374. data/app/components/hotwirebits/separator_component.rb +26 -0
  375. data/app/components/hotwirebits/sheet_component.html.erb +42 -0
  376. data/app/components/hotwirebits/sheet_component.rb +44 -0
  377. data/app/components/hotwirebits/shopping_cart_component.html.erb +46 -0
  378. data/app/components/hotwirebits/shopping_cart_component.rb +17 -0
  379. data/app/components/hotwirebits/sidebar_component.html.erb +40 -0
  380. data/app/components/hotwirebits/sidebar_component.rb +29 -0
  381. data/app/components/hotwirebits/simple_grid_component.html.erb +3 -0
  382. data/app/components/hotwirebits/simple_grid_component.rb +23 -0
  383. data/app/components/hotwirebits/skeleton_card_component.html.erb +15 -0
  384. data/app/components/hotwirebits/skeleton_card_component.rb +17 -0
  385. data/app/components/hotwirebits/skeleton_component.html.erb +5 -0
  386. data/app/components/hotwirebits/skeleton_component.rb +39 -0
  387. data/app/components/hotwirebits/slider_component.html.erb +3 -0
  388. data/app/components/hotwirebits/slider_component.rb +34 -0
  389. data/app/components/hotwirebits/social_share_component.html.erb +20 -0
  390. data/app/components/hotwirebits/social_share_component.rb +25 -0
  391. data/app/components/hotwirebits/space_component.html.erb +1 -0
  392. data/app/components/hotwirebits/space_component.rb +22 -0
  393. data/app/components/hotwirebits/speed_dial_component.html.erb +24 -0
  394. data/app/components/hotwirebits/speed_dial_component.rb +41 -0
  395. data/app/components/hotwirebits/speed_dial_fab_component.html.erb +30 -0
  396. data/app/components/hotwirebits/speed_dial_fab_component.rb +27 -0
  397. data/app/components/hotwirebits/spinner_component.html.erb +10 -0
  398. data/app/components/hotwirebits/spinner_component.rb +20 -0
  399. data/app/components/hotwirebits/split_button_component.html.erb +20 -0
  400. data/app/components/hotwirebits/split_button_component.rb +33 -0
  401. data/app/components/hotwirebits/splitter_component.html.erb +9 -0
  402. data/app/components/hotwirebits/splitter_component.rb +38 -0
  403. data/app/components/hotwirebits/spoiler_component.html.erb +17 -0
  404. data/app/components/hotwirebits/spoiler_component.rb +12 -0
  405. data/app/components/hotwirebits/stack_component.html.erb +3 -0
  406. data/app/components/hotwirebits/stack_component.rb +28 -0
  407. data/app/components/hotwirebits/stacked_list_component.html.erb +20 -0
  408. data/app/components/hotwirebits/stacked_list_component.rb +18 -0
  409. data/app/components/hotwirebits/stat_component.html.erb +21 -0
  410. data/app/components/hotwirebits/stat_component.rb +26 -0
  411. data/app/components/hotwirebits/stats_component.html.erb +15 -0
  412. data/app/components/hotwirebits/stats_component.rb +25 -0
  413. data/app/components/hotwirebits/status_bar_component.html.erb +13 -0
  414. data/app/components/hotwirebits/status_bar_component.rb +20 -0
  415. data/app/components/hotwirebits/status_component.html.erb +6 -0
  416. data/app/components/hotwirebits/status_component.rb +30 -0
  417. data/app/components/hotwirebits/status_dot_component.html.erb +11 -0
  418. data/app/components/hotwirebits/status_dot_component.rb +37 -0
  419. data/app/components/hotwirebits/stepper_form_component.html.erb +55 -0
  420. data/app/components/hotwirebits/stepper_form_component.rb +23 -0
  421. data/app/components/hotwirebits/steps_component.html.erb +45 -0
  422. data/app/components/hotwirebits/steps_component.rb +44 -0
  423. data/app/components/hotwirebits/swap_component.html.erb +18 -0
  424. data/app/components/hotwirebits/swap_component.rb +29 -0
  425. data/app/components/hotwirebits/switch_component.html.erb +9 -0
  426. data/app/components/hotwirebits/switch_component.rb +51 -0
  427. data/app/components/hotwirebits/table_component.html.erb +5 -0
  428. data/app/components/hotwirebits/table_component.rb +27 -0
  429. data/app/components/hotwirebits/table_of_contents_component.html.erb +12 -0
  430. data/app/components/hotwirebits/table_of_contents_component.rb +23 -0
  431. data/app/components/hotwirebits/tabs_component.html.erb +43 -0
  432. data/app/components/hotwirebits/tabs_component.rb +58 -0
  433. data/app/components/hotwirebits/tag_component.html.erb +3 -0
  434. data/app/components/hotwirebits/tag_component.rb +35 -0
  435. data/app/components/hotwirebits/tag_input_component.html.erb +14 -0
  436. data/app/components/hotwirebits/tag_input_component.rb +56 -0
  437. data/app/components/hotwirebits/team_section_component.html.erb +39 -0
  438. data/app/components/hotwirebits/team_section_component.rb +20 -0
  439. data/app/components/hotwirebits/terminal_component.html.erb +21 -0
  440. data/app/components/hotwirebits/terminal_component.rb +11 -0
  441. data/app/components/hotwirebits/testimonial_component.html.erb +29 -0
  442. data/app/components/hotwirebits/testimonial_component.rb +22 -0
  443. data/app/components/hotwirebits/testimonials_section_component.html.erb +34 -0
  444. data/app/components/hotwirebits/testimonials_section_component.rb +23 -0
  445. data/app/components/hotwirebits/text_gradient_component.html.erb +3 -0
  446. data/app/components/hotwirebits/text_gradient_component.rb +18 -0
  447. data/app/components/hotwirebits/text_rotate_component.html.erb +7 -0
  448. data/app/components/hotwirebits/text_rotate_component.rb +24 -0
  449. data/app/components/hotwirebits/textarea_component.html.erb +1 -0
  450. data/app/components/hotwirebits/textarea_component.rb +35 -0
  451. data/app/components/hotwirebits/theme_controller_component.html.erb +13 -0
  452. data/app/components/hotwirebits/theme_controller_component.rb +11 -0
  453. data/app/components/hotwirebits/theme_icon_component.html.erb +3 -0
  454. data/app/components/hotwirebits/theme_icon_component.rb +18 -0
  455. data/app/components/hotwirebits/theme_selector_component.html.erb +27 -0
  456. data/app/components/hotwirebits/theme_selector_component.rb +11 -0
  457. data/app/components/hotwirebits/time_picker_component.html.erb +29 -0
  458. data/app/components/hotwirebits/time_picker_component.rb +28 -0
  459. data/app/components/hotwirebits/timeline_component.html.erb +24 -0
  460. data/app/components/hotwirebits/timeline_component.rb +17 -0
  461. data/app/components/hotwirebits/toast_component.html.erb +34 -0
  462. data/app/components/hotwirebits/toast_component.rb +47 -0
  463. data/app/components/hotwirebits/toggle_button_component.html.erb +4 -0
  464. data/app/components/hotwirebits/toggle_button_component.rb +31 -0
  465. data/app/components/hotwirebits/toggle_button_group_component.html.erb +8 -0
  466. data/app/components/hotwirebits/toggle_button_group_component.rb +54 -0
  467. data/app/components/hotwirebits/toggle_component.html.erb +3 -0
  468. data/app/components/hotwirebits/toggle_component.rb +38 -0
  469. data/app/components/hotwirebits/toolbar_component.html.erb +18 -0
  470. data/app/components/hotwirebits/toolbar_component.rb +50 -0
  471. data/app/components/hotwirebits/tooltip_component.html.erb +16 -0
  472. data/app/components/hotwirebits/tooltip_component.rb +32 -0
  473. data/app/components/hotwirebits/tree_view_component.html.erb +7 -0
  474. data/app/components/hotwirebits/tree_view_component.rb +47 -0
  475. data/app/components/hotwirebits/two_factor_component.html.erb +26 -0
  476. data/app/components/hotwirebits/two_factor_component.rb +14 -0
  477. data/app/components/hotwirebits/typing_indicator_component.html.erb +5 -0
  478. data/app/components/hotwirebits/typing_indicator_component.rb +33 -0
  479. data/app/components/hotwirebits/typography_component.html.erb +3 -0
  480. data/app/components/hotwirebits/typography_component.rb +35 -0
  481. data/app/components/hotwirebits/validator_component.html.erb +20 -0
  482. data/app/components/hotwirebits/validator_component.rb +12 -0
  483. data/app/components/hotwirebits/version_diff_component.html.erb +17 -0
  484. data/app/components/hotwirebits/version_diff_component.rb +38 -0
  485. data/app/components/hotwirebits/video_player_component.html.erb +11 -0
  486. data/app/components/hotwirebits/video_player_component.rb +29 -0
  487. data/app/components/hotwirebits/voice_recorder_component.html.erb +29 -0
  488. data/app/components/hotwirebits/voice_recorder_component.rb +14 -0
  489. data/app/components/hotwirebits/window_mockup_component.html.erb +20 -0
  490. data/app/components/hotwirebits/window_mockup_component.rb +11 -0
  491. data/app/helpers/hotwirebits_helper.rb +738 -0
  492. data/app/javascript/controllers/hotwirebits/index.js +216 -0
  493. data/app/javascript/controllers/hotwirebits/rb_accordion_controller.js +16 -0
  494. data/app/javascript/controllers/hotwirebits/rb_activity_feed_controller.js +11 -0
  495. data/app/javascript/controllers/hotwirebits/rb_ai_chat_controller.js +21 -0
  496. data/app/javascript/controllers/hotwirebits/rb_alert_controller.js +24 -0
  497. data/app/javascript/controllers/hotwirebits/rb_angle_slider_controller.js +11 -0
  498. data/app/javascript/controllers/hotwirebits/rb_animated_number_controller.js +7 -0
  499. data/app/javascript/controllers/hotwirebits/rb_autocomplete_controller.js +34 -0
  500. data/app/javascript/controllers/hotwirebits/rb_badge_controller.js +8 -0
  501. data/app/javascript/controllers/hotwirebits/rb_banner_controller.js +27 -0
  502. data/app/javascript/controllers/hotwirebits/rb_burger_controller.js +25 -0
  503. data/app/javascript/controllers/hotwirebits/rb_calendar_controller.js +85 -0
  504. data/app/javascript/controllers/hotwirebits/rb_carousel_controller.js +74 -0
  505. data/app/javascript/controllers/hotwirebits/rb_clipboard_controller.js +42 -0
  506. data/app/javascript/controllers/hotwirebits/rb_code_highlight_controller.js +7 -0
  507. data/app/javascript/controllers/hotwirebits/rb_collapsible_controller.js +51 -0
  508. data/app/javascript/controllers/hotwirebits/rb_color_input_controller.js +7 -0
  509. data/app/javascript/controllers/hotwirebits/rb_color_picker_controller.js +7 -0
  510. data/app/javascript/controllers/hotwirebits/rb_combobox_controller.js +91 -0
  511. data/app/javascript/controllers/hotwirebits/rb_command_bar_controller.js +32 -0
  512. data/app/javascript/controllers/hotwirebits/rb_compare_controller.js +18 -0
  513. data/app/javascript/controllers/hotwirebits/rb_confirm_dialog_controller.js +11 -0
  514. data/app/javascript/controllers/hotwirebits/rb_confirm_popup_controller.js +10 -0
  515. data/app/javascript/controllers/hotwirebits/rb_context_menu_controller.js +27 -0
  516. data/app/javascript/controllers/hotwirebits/rb_cookie_consent_controller.js +60 -0
  517. data/app/javascript/controllers/hotwirebits/rb_cookie_settings_controller.js +9 -0
  518. data/app/javascript/controllers/hotwirebits/rb_countdown_controller.js +60 -0
  519. data/app/javascript/controllers/hotwirebits/rb_data_view_controller.js +10 -0
  520. data/app/javascript/controllers/hotwirebits/rb_date_range_controller.js +9 -0
  521. data/app/javascript/controllers/hotwirebits/rb_datepicker_controller.js +107 -0
  522. data/app/javascript/controllers/hotwirebits/rb_dialog_controller.js +68 -0
  523. data/app/javascript/controllers/hotwirebits/rb_drag_drop_controller.js +21 -0
  524. data/app/javascript/controllers/hotwirebits/rb_drawer_controller.js +69 -0
  525. data/app/javascript/controllers/hotwirebits/rb_dropdown_controller.js +73 -0
  526. data/app/javascript/controllers/hotwirebits/rb_feedback_controller.js +50 -0
  527. data/app/javascript/controllers/hotwirebits/rb_file_browser_controller.js +9 -0
  528. data/app/javascript/controllers/hotwirebits/rb_highlight_controller.js +47 -0
  529. data/app/javascript/controllers/hotwirebits/rb_hover_3d_controller.js +7 -0
  530. data/app/javascript/controllers/hotwirebits/rb_hover_card_controller.js +19 -0
  531. data/app/javascript/controllers/hotwirebits/rb_hover_gallery_controller.js +7 -0
  532. data/app/javascript/controllers/hotwirebits/rb_image_compare_controller.js +7 -0
  533. data/app/javascript/controllers/hotwirebits/rb_image_controller.js +7 -0
  534. data/app/javascript/controllers/hotwirebits/rb_image_zoom_controller.js +7 -0
  535. data/app/javascript/controllers/hotwirebits/rb_inline_edit_controller.js +7 -0
  536. data/app/javascript/controllers/hotwirebits/rb_inplace_controller.js +7 -0
  537. data/app/javascript/controllers/hotwirebits/rb_input_number_controller.js +9 -0
  538. data/app/javascript/controllers/hotwirebits/rb_json_input_controller.js +7 -0
  539. data/app/javascript/controllers/hotwirebits/rb_kanban_controller.js +78 -0
  540. data/app/javascript/controllers/hotwirebits/rb_knob_controller.js +7 -0
  541. data/app/javascript/controllers/hotwirebits/rb_lightbox_controller.js +36 -0
  542. data/app/javascript/controllers/hotwirebits/rb_like_controller.js +7 -0
  543. data/app/javascript/controllers/hotwirebits/rb_listbox_controller.js +7 -0
  544. data/app/javascript/controllers/hotwirebits/rb_marquee_controller.js +68 -0
  545. data/app/javascript/controllers/hotwirebits/rb_menubar_controller.js +78 -0
  546. data/app/javascript/controllers/hotwirebits/rb_month_picker_controller.js +7 -0
  547. data/app/javascript/controllers/hotwirebits/rb_month_picker_input_controller.js +7 -0
  548. data/app/javascript/controllers/hotwirebits/rb_multiselect_controller.js +83 -0
  549. data/app/javascript/controllers/hotwirebits/rb_nav_menu_controller.js +7 -0
  550. data/app/javascript/controllers/hotwirebits/rb_newsletter_controller.js +7 -0
  551. data/app/javascript/controllers/hotwirebits/rb_notification_controller.js +7 -0
  552. data/app/javascript/controllers/hotwirebits/rb_otp_controller.js +54 -0
  553. data/app/javascript/controllers/hotwirebits/rb_paginator_controller.js +7 -0
  554. data/app/javascript/controllers/hotwirebits/rb_panel_controller.js +12 -0
  555. data/app/javascript/controllers/hotwirebits/rb_password_controller.js +36 -0
  556. data/app/javascript/controllers/hotwirebits/rb_password_strength_controller.js +7 -0
  557. data/app/javascript/controllers/hotwirebits/rb_password_toggle_controller.js +7 -0
  558. data/app/javascript/controllers/hotwirebits/rb_popover_controller.js +61 -0
  559. data/app/javascript/controllers/hotwirebits/rb_portal_controller.js +7 -0
  560. data/app/javascript/controllers/hotwirebits/rb_quickview_controller.js +7 -0
  561. data/app/javascript/controllers/hotwirebits/rb_range_slider_controller.js +7 -0
  562. data/app/javascript/controllers/hotwirebits/rb_rating_controller.js +55 -0
  563. data/app/javascript/controllers/hotwirebits/rb_resizable_controller.js +82 -0
  564. data/app/javascript/controllers/hotwirebits/rb_rich_text_editor_controller.js +7 -0
  565. data/app/javascript/controllers/hotwirebits/rb_search_controller.js +70 -0
  566. data/app/javascript/controllers/hotwirebits/rb_sheet_controller.js +43 -0
  567. data/app/javascript/controllers/hotwirebits/rb_sidebar_controller.js +66 -0
  568. data/app/javascript/controllers/hotwirebits/rb_signature_controller.js +7 -0
  569. data/app/javascript/controllers/hotwirebits/rb_speed_dial_controller.js +64 -0
  570. data/app/javascript/controllers/hotwirebits/rb_splitter_controller.js +27 -0
  571. data/app/javascript/controllers/hotwirebits/rb_spoiler_controller.js +26 -0
  572. data/app/javascript/controllers/hotwirebits/rb_stepper_form_controller.js +11 -0
  573. data/app/javascript/controllers/hotwirebits/rb_steps_controller.js +46 -0
  574. data/app/javascript/controllers/hotwirebits/rb_swap_controller.js +22 -0
  575. data/app/javascript/controllers/hotwirebits/rb_switch_controller.js +33 -0
  576. data/app/javascript/controllers/hotwirebits/rb_tabs_controller.js +46 -0
  577. data/app/javascript/controllers/hotwirebits/rb_tag_input_controller.js +19 -0
  578. data/app/javascript/controllers/hotwirebits/rb_tags_controller.js +50 -0
  579. data/app/javascript/controllers/hotwirebits/rb_text_rotate_controller.js +7 -0
  580. data/app/javascript/controllers/hotwirebits/rb_theme_controller.js +48 -0
  581. data/app/javascript/controllers/hotwirebits/rb_theme_selector_controller.js +7 -0
  582. data/app/javascript/controllers/hotwirebits/rb_time_picker_controller.js +7 -0
  583. data/app/javascript/controllers/hotwirebits/rb_toast_controller.js +43 -0
  584. data/app/javascript/controllers/hotwirebits/rb_toggle_button_controller.js +9 -0
  585. data/app/javascript/controllers/hotwirebits/rb_toggle_button_group_controller.js +12 -0
  586. data/app/javascript/controllers/hotwirebits/rb_toggle_controller.js +28 -0
  587. data/app/javascript/controllers/hotwirebits/rb_tooltip_controller.js +7 -0
  588. data/app/javascript/controllers/hotwirebits/rb_tree_controller.js +58 -0
  589. data/app/javascript/controllers/hotwirebits/rb_two_factor_controller.js +7 -0
  590. data/app/javascript/controllers/hotwirebits/rb_validator_controller.js +7 -0
  591. data/app/javascript/controllers/hotwirebits/rb_voice_recorder_controller.js +7 -0
  592. data/app/views/hotwirebits/_accordion.html.erb +26 -0
  593. data/app/views/hotwirebits/_activity_feed.html.erb +43 -0
  594. data/app/views/hotwirebits/_affix.html.erb +6 -0
  595. data/app/views/hotwirebits/_ai_chat.html.erb +35 -0
  596. data/app/views/hotwirebits/_alert.html.erb +53 -0
  597. data/app/views/hotwirebits/_alert_dialog.html.erb +33 -0
  598. data/app/views/hotwirebits/_anchor.html.erb +12 -0
  599. data/app/views/hotwirebits/_angle_slider.html.erb +12 -0
  600. data/app/views/hotwirebits/_animated_number.html.erb +11 -0
  601. data/app/views/hotwirebits/_app_shell.html.erb +25 -0
  602. data/app/views/hotwirebits/_aspect_ratio.html.erb +6 -0
  603. data/app/views/hotwirebits/_auth_block.html.erb +66 -0
  604. data/app/views/hotwirebits/_autocomplete.html.erb +8 -0
  605. data/app/views/hotwirebits/_avatar.html.erb +32 -0
  606. data/app/views/hotwirebits/_background_image.html.erb +12 -0
  607. data/app/views/hotwirebits/_badge.html.erb +32 -0
  608. data/app/views/hotwirebits/_banner.html.erb +52 -0
  609. data/app/views/hotwirebits/_bento_grid.html.erb +10 -0
  610. data/app/views/hotwirebits/_blog_section.html.erb +38 -0
  611. data/app/views/hotwirebits/_bottom_navigation.html.erb +24 -0
  612. data/app/views/hotwirebits/_box.html.erb +6 -0
  613. data/app/views/hotwirebits/_breadcrumb.html.erb +28 -0
  614. data/app/views/hotwirebits/_breadcrumb_advanced.html.erb +56 -0
  615. data/app/views/hotwirebits/_browser_mockup.html.erb +25 -0
  616. data/app/views/hotwirebits/_burger.html.erb +23 -0
  617. data/app/views/hotwirebits/_button.html.erb +59 -0
  618. data/app/views/hotwirebits/_calendar.html.erb +73 -0
  619. data/app/views/hotwirebits/_card.html.erb +23 -0
  620. data/app/views/hotwirebits/_card_heading.html.erb +16 -0
  621. data/app/views/hotwirebits/_carousel.html.erb +53 -0
  622. data/app/views/hotwirebits/_center.html.erb +6 -0
  623. data/app/views/hotwirebits/_chart.html.erb +71 -0
  624. data/app/views/hotwirebits/_chat_bubble.html.erb +30 -0
  625. data/app/views/hotwirebits/_checkbox.html.erb +25 -0
  626. data/app/views/hotwirebits/_checkout_form.html.erb +43 -0
  627. data/app/views/hotwirebits/_chip.html.erb +30 -0
  628. data/app/views/hotwirebits/_clipboard.html.erb +29 -0
  629. data/app/views/hotwirebits/_close_button.html.erb +6 -0
  630. data/app/views/hotwirebits/_code_block.html.erb +20 -0
  631. data/app/views/hotwirebits/_code_highlight.html.erb +9 -0
  632. data/app/views/hotwirebits/_code_mockup.html.erb +16 -0
  633. data/app/views/hotwirebits/_collapsible.html.erb +20 -0
  634. data/app/views/hotwirebits/_color_input.html.erb +22 -0
  635. data/app/views/hotwirebits/_color_picker.html.erb +9 -0
  636. data/app/views/hotwirebits/_color_swatch.html.erb +4 -0
  637. data/app/views/hotwirebits/_combobox.html.erb +18 -0
  638. data/app/views/hotwirebits/_command_bar.html.erb +36 -0
  639. data/app/views/hotwirebits/_compare.html.erb +24 -0
  640. data/app/views/hotwirebits/_confirm_dialog.html.erb +20 -0
  641. data/app/views/hotwirebits/_confirm_popup.html.erb +19 -0
  642. data/app/views/hotwirebits/_contact_section.html.erb +56 -0
  643. data/app/views/hotwirebits/_container.html.erb +21 -0
  644. data/app/views/hotwirebits/_context_menu.html.erb +30 -0
  645. data/app/views/hotwirebits/_cookie_consent.html.erb +26 -0
  646. data/app/views/hotwirebits/_cookie_settings.html.erb +51 -0
  647. data/app/views/hotwirebits/_countdown.html.erb +11 -0
  648. data/app/views/hotwirebits/_cta_section.html.erb +36 -0
  649. data/app/views/hotwirebits/_dashboard_card.html.erb +35 -0
  650. data/app/views/hotwirebits/_data_table.html.erb +56 -0
  651. data/app/views/hotwirebits/_data_view.html.erb +24 -0
  652. data/app/views/hotwirebits/_date_picker.html.erb +11 -0
  653. data/app/views/hotwirebits/_date_range_picker.html.erb +31 -0
  654. data/app/views/hotwirebits/_device_frame.html.erb +43 -0
  655. data/app/views/hotwirebits/_device_mockup.html.erb +54 -0
  656. data/app/views/hotwirebits/_dialog.html.erb +58 -0
  657. data/app/views/hotwirebits/_diff.html.erb +25 -0
  658. data/app/views/hotwirebits/_diff_view.html.erb +25 -0
  659. data/app/views/hotwirebits/_digital_signature.html.erb +24 -0
  660. data/app/views/hotwirebits/_divider.html.erb +22 -0
  661. data/app/views/hotwirebits/_dock_menu.html.erb +24 -0
  662. data/app/views/hotwirebits/_drag_drop_list.html.erb +19 -0
  663. data/app/views/hotwirebits/_drawer.html.erb +41 -0
  664. data/app/views/hotwirebits/_dropdown.html.erb +33 -0
  665. data/app/views/hotwirebits/_dropzone.html.erb +19 -0
  666. data/app/views/hotwirebits/_empty_cart.html.erb +22 -0
  667. data/app/views/hotwirebits/_empty_state.html.erb +29 -0
  668. data/app/views/hotwirebits/_error_page.html.erb +17 -0
  669. data/app/views/hotwirebits/_faq_section.html.erb +29 -0
  670. data/app/views/hotwirebits/_feature_section.html.erb +30 -0
  671. data/app/views/hotwirebits/_feedback.html.erb +54 -0
  672. data/app/views/hotwirebits/_fieldset.html.erb +14 -0
  673. data/app/views/hotwirebits/_file_browser.html.erb +38 -0
  674. data/app/views/hotwirebits/_file_input.html.erb +10 -0
  675. data/app/views/hotwirebits/_file_upload_progress.html.erb +29 -0
  676. data/app/views/hotwirebits/_flex.html.erb +6 -0
  677. data/app/views/hotwirebits/_float_label.html.erb +7 -0
  678. data/app/views/hotwirebits/_fluid.html.erb +6 -0
  679. data/app/views/hotwirebits/_flyout_menu.html.erb +38 -0
  680. data/app/views/hotwirebits/_footer.html.erb +46 -0
  681. data/app/views/hotwirebits/_form_group.html.erb +43 -0
  682. data/app/views/hotwirebits/_gallery.html.erb +28 -0
  683. data/app/views/hotwirebits/_grid.html.erb +14 -0
  684. data/app/views/hotwirebits/_group.html.erb +6 -0
  685. data/app/views/hotwirebits/_header.html.erb +28 -0
  686. data/app/views/hotwirebits/_header_section.html.erb +36 -0
  687. data/app/views/hotwirebits/_hero.html.erb +38 -0
  688. data/app/views/hotwirebits/_highlight.html.erb +4 -0
  689. data/app/views/hotwirebits/_hover_3d_card.html.erb +13 -0
  690. data/app/views/hotwirebits/_hover_card.html.erb +22 -0
  691. data/app/views/hotwirebits/_hover_gallery.html.erb +26 -0
  692. data/app/views/hotwirebits/_icon_field.html.erb +12 -0
  693. data/app/views/hotwirebits/_ifta_label.html.erb +7 -0
  694. data/app/views/hotwirebits/_image.html.erb +22 -0
  695. data/app/views/hotwirebits/_image_compare.html.erb +24 -0
  696. data/app/views/hotwirebits/_image_zoom.html.erb +11 -0
  697. data/app/views/hotwirebits/_incentive.html.erb +14 -0
  698. data/app/views/hotwirebits/_indicator.html.erb +9 -0
  699. data/app/views/hotwirebits/_indicator_badge.html.erb +11 -0
  700. data/app/views/hotwirebits/_inline_edit.html.erb +53 -0
  701. data/app/views/hotwirebits/_inplace.html.erb +9 -0
  702. data/app/views/hotwirebits/_input.html.erb +33 -0
  703. data/app/views/hotwirebits/_input_group.html.erb +12 -0
  704. data/app/views/hotwirebits/_input_number.html.erb +17 -0
  705. data/app/views/hotwirebits/_input_otp.html.erb +9 -0
  706. data/app/views/hotwirebits/_input_tags.html.erb +15 -0
  707. data/app/views/hotwirebits/_join.html.erb +6 -0
  708. data/app/views/hotwirebits/_json_input.html.erb +17 -0
  709. data/app/views/hotwirebits/_kanban_board.html.erb +42 -0
  710. data/app/views/hotwirebits/_kbd.html.erb +17 -0
  711. data/app/views/hotwirebits/_knob.html.erb +19 -0
  712. data/app/views/hotwirebits/_label.html.erb +8 -0
  713. data/app/views/hotwirebits/_lightbox.html.erb +33 -0
  714. data/app/views/hotwirebits/_like_button.html.erb +18 -0
  715. data/app/views/hotwirebits/_list.html.erb +6 -0
  716. data/app/views/hotwirebits/_listbox.html.erb +19 -0
  717. data/app/views/hotwirebits/_loading_overlay.html.erb +15 -0
  718. data/app/views/hotwirebits/_logo_cloud.html.erb +23 -0
  719. data/app/views/hotwirebits/_logo_item.html.erb +10 -0
  720. data/app/views/hotwirebits/_mark.html.erb +4 -0
  721. data/app/views/hotwirebits/_marquee.html.erb +32 -0
  722. data/app/views/hotwirebits/_mask.html.erb +6 -0
  723. data/app/views/hotwirebits/_mega_menu.html.erb +46 -0
  724. data/app/views/hotwirebits/_menu.html.erb +35 -0
  725. data/app/views/hotwirebits/_menubar.html.erb +33 -0
  726. data/app/views/hotwirebits/_meter_group.html.erb +19 -0
  727. data/app/views/hotwirebits/_month_picker.html.erb +28 -0
  728. data/app/views/hotwirebits/_month_picker_input.html.erb +23 -0
  729. data/app/views/hotwirebits/_multiselect.html.erb +21 -0
  730. data/app/views/hotwirebits/_native_select.html.erb +13 -0
  731. data/app/views/hotwirebits/_nav_link.html.erb +12 -0
  732. data/app/views/hotwirebits/_navbar.html.erb +30 -0
  733. data/app/views/hotwirebits/_navigation_menu.html.erb +46 -0
  734. data/app/views/hotwirebits/_newsletter_section.html.erb +21 -0
  735. data/app/views/hotwirebits/_notification.html.erb +31 -0
  736. data/app/views/hotwirebits/_notification_bell.html.erb +21 -0
  737. data/app/views/hotwirebits/_number_formatter.html.erb +4 -0
  738. data/app/views/hotwirebits/_number_input.html.erb +4 -0
  739. data/app/views/hotwirebits/_order_summary.html.erb +40 -0
  740. data/app/views/hotwirebits/_org_chart.html.erb +37 -0
  741. data/app/views/hotwirebits/_page_heading.html.erb +31 -0
  742. data/app/views/hotwirebits/_pagination.html.erb +46 -0
  743. data/app/views/hotwirebits/_paginator.html.erb +39 -0
  744. data/app/views/hotwirebits/_panel.html.erb +17 -0
  745. data/app/views/hotwirebits/_password_input.html.erb +11 -0
  746. data/app/views/hotwirebits/_password_strength.html.erb +21 -0
  747. data/app/views/hotwirebits/_password_toggle.html.erb +18 -0
  748. data/app/views/hotwirebits/_permission_badge.html.erb +18 -0
  749. data/app/views/hotwirebits/_phone_input.html.erb +4 -0
  750. data/app/views/hotwirebits/_phone_mockup.html.erb +15 -0
  751. data/app/views/hotwirebits/_popover.html.erb +23 -0
  752. data/app/views/hotwirebits/_portal.html.erb +6 -0
  753. data/app/views/hotwirebits/_pricing.html.erb +55 -0
  754. data/app/views/hotwirebits/_pricing_section.html.erb +49 -0
  755. data/app/views/hotwirebits/_product_list.html.erb +27 -0
  756. data/app/views/hotwirebits/_product_overview.html.erb +62 -0
  757. data/app/views/hotwirebits/_product_quickview.html.erb +48 -0
  758. data/app/views/hotwirebits/_progress.html.erb +13 -0
  759. data/app/views/hotwirebits/_progress_ring.html.erb +29 -0
  760. data/app/views/hotwirebits/_promo_section.html.erb +34 -0
  761. data/app/views/hotwirebits/_qr_code.html.erb +22 -0
  762. data/app/views/hotwirebits/_radio.html.erb +9 -0
  763. data/app/views/hotwirebits/_range_slider.html.erb +12 -0
  764. data/app/views/hotwirebits/_rating.html.erb +14 -0
  765. data/app/views/hotwirebits/_resizable.html.erb +22 -0
  766. data/app/views/hotwirebits/_review.html.erb +39 -0
  767. data/app/views/hotwirebits/_rich_text_editor.html.erb +58 -0
  768. data/app/views/hotwirebits/_scroll_area.html.erb +6 -0
  769. data/app/views/hotwirebits/_search.html.erb +59 -0
  770. data/app/views/hotwirebits/_search_input.html.erb +9 -0
  771. data/app/views/hotwirebits/_section_heading.html.erb +9 -0
  772. data/app/views/hotwirebits/_select.html.erb +30 -0
  773. data/app/views/hotwirebits/_semi_circle_progress.html.erb +12 -0
  774. data/app/views/hotwirebits/_separator.html.erb +18 -0
  775. data/app/views/hotwirebits/_sheet.html.erb +45 -0
  776. data/app/views/hotwirebits/_shopping_cart.html.erb +49 -0
  777. data/app/views/hotwirebits/_sidebar.html.erb +39 -0
  778. data/app/views/hotwirebits/_simple_grid.html.erb +6 -0
  779. data/app/views/hotwirebits/_skeleton.html.erb +20 -0
  780. data/app/views/hotwirebits/_skeleton_card.html.erb +18 -0
  781. data/app/views/hotwirebits/_slider.html.erb +17 -0
  782. data/app/views/hotwirebits/_social_share.html.erb +23 -0
  783. data/app/views/hotwirebits/_space.html.erb +4 -0
  784. data/app/views/hotwirebits/_speed_dial.html.erb +27 -0
  785. data/app/views/hotwirebits/_speed_dial_fab.html.erb +33 -0
  786. data/app/views/hotwirebits/_spinner.html.erb +22 -0
  787. data/app/views/hotwirebits/_split_button.html.erb +23 -0
  788. data/app/views/hotwirebits/_splitter.html.erb +12 -0
  789. data/app/views/hotwirebits/_spoiler.html.erb +20 -0
  790. data/app/views/hotwirebits/_stack.html.erb +20 -0
  791. data/app/views/hotwirebits/_stacked_list.html.erb +23 -0
  792. data/app/views/hotwirebits/_stat.html.erb +24 -0
  793. data/app/views/hotwirebits/_stats.html.erb +6 -0
  794. data/app/views/hotwirebits/_status.html.erb +9 -0
  795. data/app/views/hotwirebits/_status_bar.html.erb +16 -0
  796. data/app/views/hotwirebits/_status_dot.html.erb +9 -0
  797. data/app/views/hotwirebits/_stepper_form.html.erb +58 -0
  798. data/app/views/hotwirebits/_steps.html.erb +48 -0
  799. data/app/views/hotwirebits/_swap.html.erb +20 -0
  800. data/app/views/hotwirebits/_switch.html.erb +19 -0
  801. data/app/views/hotwirebits/_table.html.erb +8 -0
  802. data/app/views/hotwirebits/_table_of_contents.html.erb +15 -0
  803. data/app/views/hotwirebits/_tabs.html.erb +44 -0
  804. data/app/views/hotwirebits/_tag.html.erb +20 -0
  805. data/app/views/hotwirebits/_tag_input.html.erb +17 -0
  806. data/app/views/hotwirebits/_team_section.html.erb +42 -0
  807. data/app/views/hotwirebits/_terminal.html.erb +24 -0
  808. data/app/views/hotwirebits/_testimonial.html.erb +32 -0
  809. data/app/views/hotwirebits/_testimonials_section.html.erb +37 -0
  810. data/app/views/hotwirebits/_text_gradient.html.erb +6 -0
  811. data/app/views/hotwirebits/_text_rotate.html.erb +10 -0
  812. data/app/views/hotwirebits/_textarea.html.erb +17 -0
  813. data/app/views/hotwirebits/_theme_controller.html.erb +16 -0
  814. data/app/views/hotwirebits/_theme_icon.html.erb +6 -0
  815. data/app/views/hotwirebits/_theme_selector.html.erb +30 -0
  816. data/app/views/hotwirebits/_time_picker.html.erb +32 -0
  817. data/app/views/hotwirebits/_timeline.html.erb +27 -0
  818. data/app/views/hotwirebits/_toast.html.erb +52 -0
  819. data/app/views/hotwirebits/_toggle.html.erb +12 -0
  820. data/app/views/hotwirebits/_toggle_button.html.erb +7 -0
  821. data/app/views/hotwirebits/_toggle_button_group.html.erb +11 -0
  822. data/app/views/hotwirebits/_toolbar.html.erb +21 -0
  823. data/app/views/hotwirebits/_tooltip.html.erb +26 -0
  824. data/app/views/hotwirebits/_tree_view.html.erb +18 -0
  825. data/app/views/hotwirebits/_two_factor.html.erb +29 -0
  826. data/app/views/hotwirebits/_typing_indicator.html.erb +8 -0
  827. data/app/views/hotwirebits/_typography.html.erb +7 -0
  828. data/app/views/hotwirebits/_validator.html.erb +23 -0
  829. data/app/views/hotwirebits/_version_diff.html.erb +20 -0
  830. data/app/views/hotwirebits/_video_player.html.erb +14 -0
  831. data/app/views/hotwirebits/_voice_recorder.html.erb +32 -0
  832. data/app/views/hotwirebits/_window_mockup.html.erb +23 -0
  833. data/config/routes.rb +4 -0
  834. data/lib/generators/hotwirebits/component_generator.rb +48 -0
  835. data/lib/generators/hotwirebits/install_generator.rb +47 -0
  836. data/lib/generators/hotwirebits/templates/component.html.erb.tt +4 -0
  837. data/lib/generators/hotwirebits/templates/component.rb.tt +9 -0
  838. data/lib/generators/hotwirebits/templates/controller.js.tt +6 -0
  839. data/lib/generators/hotwirebits/templates/theme.css +2 -0
  840. data/lib/hotwirebits/engine.rb +25 -0
  841. data/lib/hotwirebits/version.rb +5 -0
  842. data/lib/hotwirebits.rb +19 -0
  843. metadata +965 -0
@@ -0,0 +1,62 @@
1
+ <%# ProductOverview component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/product_overview', ... %>
3
+
4
+ <div class="mx-auto max-w-7xl px-6 py-10 lg:px-8" <%= tag.attributes(local_assigns) %>>
5
+ <div class="grid gap-10 lg:grid-cols-2">
6
+ <div class="space-y-4">
7
+ <% if main_image %>
8
+ <div class="overflow-hidden rounded-xl bg-hw-muted">
9
+ <img src="<%= main_image[:src] %>" alt="<%= main_image[:alt] || name %>" class="w-full object-cover">
10
+ </div>
11
+ <% end %>
12
+ <% if images.size > 1 %>
13
+ <div class="flex gap-3">
14
+ <% images.each do |img| %>
15
+ <button type="button" class="h-16 w-16 overflow-hidden rounded-lg border-2 border-transparent hover:border-hw-primary">
16
+ <img src="<%= img[:src] %>" alt="<%= img[:alt] || name %>" class="h-full w-full object-cover">
17
+ </button>
18
+ <% end %>
19
+ </div>
20
+ <% end %>
21
+ </div>
22
+ <div>
23
+ <h1 class="text-2xl font-bold text-hw-foreground"><%= name %></h1>
24
+ <p class="mt-2 text-2xl font-semibold text-hw-primary"><%= price %></p>
25
+ <% if rating %>
26
+ <div class="mt-3 flex items-center gap-1">
27
+ <% 5.times do |i| %>
28
+ <svg class="h-5 w-5 <%= i < rating.to_i ? 'text-hw-warning' : 'text-hw-muted-foreground/40' %>" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
29
+ <path fill-rule="evenodd" d="M10.868 2.884c-.321-.772-1.415-.772-1.736 0l-1.83 4.401-4.753.381c-.833.067-1.171 1.107-.536 1.651l3.62 3.102-1.106 4.637c-.194.813.691 1.456 1.405 1.02L10 15.591l4.069 2.485c.713.436 1.598-.207 1.404-1.02l-1.106-4.637 3.62-3.102c.635-.544.297-1.584-.536-1.65l-4.752-.382-1.831-4.401z" clip-rule="evenodd"/>
30
+ </svg>
31
+ <% end %>
32
+ <span class="ml-1 text-sm text-hw-muted-foreground">(<%= rating %>)</span>
33
+ </div>
34
+ <% end %>
35
+ <% if description %>
36
+ <div class="mt-6 text-sm leading-relaxed text-hw-muted-foreground"><%= description %></div>
37
+ <% end %>
38
+ <% if features.any? %>
39
+ <ul class="mt-6 space-y-2">
40
+ <% features.each do |feature| %>
41
+ <li class="flex items-center gap-2 text-sm text-hw-foreground">
42
+ <svg class="h-4 w-4 text-green-500 shrink-0" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
43
+ <path fill-rule="evenodd" d="M16.704 4.153a.75.75 0 01.143 1.052l-8 10.5a.75.75 0 01-1.127.075l-4.5-4.5a.75.75 0 011.06-1.06l3.894 3.893 7.48-9.817a.75.75 0 011.05-.143z" clip-rule="evenodd"/>
44
+ </svg>
45
+ <%= feature %>
46
+ </li>
47
+ <% end %>
48
+ </ul>
49
+ <% end %>
50
+ <div class="mt-8 flex gap-3">
51
+ <button type="button" class="flex-1 rounded-lg bg-hw-primary px-6 py-3 text-sm font-semibold text-hw-primary-foreground transition-colors hover:bg-hw-primary/90">
52
+ Add to Cart
53
+ </button>
54
+ <button type="button" class="rounded-lg border border-hw-border px-4 py-3 text-sm font-medium text-hw-foreground hover:bg-hw-muted">
55
+ <svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor">
56
+ <path stroke-linecap="round" stroke-linejoin="round" d="M21 8.25c0-2.485-2.099-4.5-4.688-4.5-1.935 0-3.597 1.126-4.312 2.733-.715-1.607-2.377-2.733-4.313-2.733C5.1 3.75 3 5.765 3 8.25c0 7.22 9 12 9 12s9-4.78 9-12z"/>
57
+ </svg>
58
+ </button>
59
+ </div>
60
+ </div>
61
+ </div>
62
+ </div>
@@ -0,0 +1,48 @@
1
+ <%# ProductQuickview component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/product_quickview', ... %>
3
+
4
+ <div class="rounded-xl border border-hw-border bg-hw-card overflow-hidden shadow-lg max-w-sm"
5
+ data-controller="hw-quickview"
6
+ <%= tag.attributes(local_assigns) %>>
7
+ <% if image %>
8
+ <div class="relative aspect-square bg-hw-muted">
9
+ <img src="<%= image %>" alt="<%= name %>" class="h-full w-full object-cover">
10
+ <button type="button"
11
+ class="absolute top-3 right-3 h-8 w-8 inline-flex items-center justify-center rounded-full bg-hw-background/80 backdrop-blur text-hw-muted-foreground hover:bg-hw-background hover:text-hw-foreground transition-colors cursor-pointer"
12
+ data-action="click->hw-quickview#close"
13
+ aria-label="Close">
14
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12"/></svg>
15
+ </button>
16
+ </div>
17
+ <% end %>
18
+ <div class="p-4 space-y-3">
19
+ <div class="flex items-start justify-between gap-2">
20
+ <h3 class="text-base font-semibold text-hw-foreground"><%= name %></h3>
21
+ <span class="text-lg font-bold text-hw-primary"><%= price %></span>
22
+ </div>
23
+ <% if rating %>
24
+ <div class="flex items-center gap-1">
25
+ <% 5.times do |i| %>
26
+ <svg class="h-4 w-4 <%= i < rating ? 'text-hw-warning' : 'text-hw-muted' %>" xmlns="http://www.w3.org/2000/svg" fill="<%= i < rating ? 'currentColor' : 'none' %>" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
27
+ <path stroke-linecap="round" stroke-linejoin="round" d="M11.049 2.927c.3-.921 1.603-.921 1.902 0l1.519 4.674a1 1 0 00.95.69h4.915c.969 0 1.371 1.24.588 1.81l-3.976 2.888a1 1 0 00-.363 1.118l1.518 4.674c.3.922-.755 1.688-1.538 1.118l-3.976-2.888a1 1 0 00-1.176 0l-3.976 2.888c-.783.57-1.838-.197-1.538-1.118l1.518-4.674a1 1 0 00-.363-1.118l-3.976-2.888c-.784-.57-.38-1.81.588-1.81h4.914a1 1 0 00.951-.69l1.519-4.674z"/>
28
+ </svg>
29
+ <% end %>
30
+ </div>
31
+ <% end %>
32
+ <% if description %>
33
+ <p class="text-sm text-hw-muted-foreground line-clamp-2"><%= description %></p>
34
+ <% end %>
35
+ <div class="flex gap-2 pt-2">
36
+ <button type="button"
37
+ class="flex-1 inline-flex h-9 items-center justify-center rounded-md bg-hw-primary px-4 text-sm font-medium text-hw-primary-foreground hover:bg-hw-primary/90 transition-colors cursor-pointer">
38
+ Add to Cart
39
+ </button>
40
+ <% if href %>
41
+ <a href="<%= href %>"
42
+ class="inline-flex h-9 items-center justify-center rounded-md border border-hw-input bg-hw-background px-4 text-sm font-medium text-hw-foreground hover:bg-hw-accent transition-colors">
43
+ Details
44
+ </a>
45
+ <% end %>
46
+ </div>
47
+ </div>
48
+ </div>
@@ -0,0 +1,13 @@
1
+ <%# Progress component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/progress', ... %>
3
+
4
+ <div <%= tag.attributes(local_assigns) %>>
5
+ <div class="flex items-center justify-between mb-1">
6
+ <% if show_label %>
7
+ <span class="text-sm font-medium"><%= percentage %>%</span>
8
+ <% end %>
9
+ </div>
10
+ <div class="<%= track_classes %>">
11
+ <div class="<%= bar_classes %>" style="width: <%= percentage %>%;"></div>
12
+ </div>
13
+ </div>
@@ -0,0 +1,29 @@
1
+ <%# ProgressRing component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/progress_ring', ... %>
3
+
4
+ <div class="relative inline-flex items-center justify-center" <%= tag.attributes(local_assigns) %>>
5
+ <svg width="<%= size %>" height="<%= size %>" class="-rotate-90">
6
+ <circle
7
+ cx="<%= center %>"
8
+ cy="<%= center %>"
9
+ r="<%= radius %>"
10
+ fill="none"
11
+ stroke-width="4"
12
+ class="stroke-hw-muted"
13
+ />
14
+ <circle
15
+ cx="<%= center %>"
16
+ cy="<%= center %>"
17
+ r="<%= radius %>"
18
+ fill="none"
19
+ stroke-width="4"
20
+ stroke-linecap="round"
21
+ stroke-dasharray="<%= circumference %>"
22
+ stroke-dashoffset="<%= stroke_dashoffset %>"
23
+ class="<%= color_class %> transition-all duration-300"
24
+ />
25
+ </svg>
26
+ <% if show_label %>
27
+ <span class="absolute text-sm font-medium"><%= percentage %>%</span>
28
+ <% end %>
29
+ </div>
@@ -0,0 +1,34 @@
1
+ <%# PromoSection component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/promo_section', ... %>
3
+
4
+ <section class="<%= section_classes %>" <%= tag.attributes(local_assigns) %>>
5
+ <div class="mx-auto grid max-w-7xl items-center gap-8 px-6 py-16 lg:grid-cols-2 lg:gap-16 lg:px-8">
6
+ <div class="flex flex-col items-start gap-4">
7
+ <% if badge %>
8
+ <span class="inline-flex items-center rounded-full px-3 py-1 text-xs font-semibold ring-1 ring-inset
9
+ <%= variant == :dark || variant == :gradient ? 'ring-white/30 text-hw-primary-foreground/90' : 'ring-hw-primary/30 text-hw-primary bg-hw-primary/5' %>">
10
+ <%= badge %>
11
+ </span>
12
+ <% end %>
13
+ <h2 class="text-3xl font-bold tracking-tight sm:text-4xl"><%= title %></h2>
14
+ <% if description %>
15
+ <p class="text-lg opacity-80"><%= description %></p>
16
+ <% end %>
17
+ <% if cta %>
18
+ <a href="<%= cta[:href] %>"
19
+ class="mt-2 inline-flex items-center rounded-lg px-5 py-2.5 text-sm font-semibold transition-colors
20
+ <%= variant == :dark || variant == :gradient ? 'bg-white text-hw-foreground hover:bg-white/90' : 'bg-hw-primary text-hw-primary-foreground hover:bg-hw-primary/90' %>">
21
+ <%= cta[:label] %>
22
+ <svg class="ml-2 h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor">
23
+ <path stroke-linecap="round" stroke-linejoin="round" d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3"/>
24
+ </svg>
25
+ </a>
26
+ <% end %>
27
+ </div>
28
+ <% if image %>
29
+ <div class="relative">
30
+ <img src="<%= image %>" alt="" class="w-full rounded-xl object-cover shadow-2xl">
31
+ </div>
32
+ <% end %>
33
+ </div>
34
+ </section>
@@ -0,0 +1,22 @@
1
+ <%# QrCode component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/qr_code', ... %>
3
+
4
+ <% qr = modules %>
5
+ <% cell_size = (size.to_f / (qr.size + margin * 2)).round(2) %>
6
+ <div class="<%= merge_classes('inline-block', extra_attrs.delete(:class)) %>"
7
+ <%= tag.attributes(local_assigns) %>>
8
+ <svg xmlns="http://www.w3.org/2000/svg"
9
+ width="<%= size %>"
10
+ height="<%= size %>"
11
+ viewBox="0 0 <%= qr.size + margin * 2 %> <%= qr.size + margin * 2 %>"
12
+ shape-rendering="crispEdges">
13
+ <rect width="<%= qr.size + margin * 2 %>" height="<%= qr.size + margin * 2 %>" fill="white"/>
14
+ <% qr.each_with_index do |row, r| %>
15
+ <% row.each_with_index do |cell, c| %>
16
+ <% if cell == 1 %>
17
+ <rect x="<%= c + margin %>" y="<%= r + margin %>" width="1" height="1" fill="black"/>
18
+ <% end %>
19
+ <% end %>
20
+ <% end %>
21
+ </svg>
22
+ </div>
@@ -0,0 +1,9 @@
1
+ <%# Radio component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/radio', ... %>
3
+
4
+ <label class="flex items-center gap-2 cursor-pointer" <%= tag.attributes(local_assigns) %>>
5
+ <input type="radio" name="<%= name %>" value="<%= value %>" <%= 'checked' if checked %> <%= 'disabled' if disabled %> class="peer aspect-square shrink-0 rounded-full border border-hw-input bg-hw-background text-hw-primary ring-offset-hw-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-hw-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50 <%= SIZES[size] || 'h-4 w-4' %>">
6
+ <% if label %>
7
+ <span class="text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70"><%= label %></span>
8
+ <% end %>
9
+ </label>
@@ -0,0 +1,12 @@
1
+ <%# RangeSlider component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/range_slider', ... %>
3
+
4
+ <% left_pct = ((value_min - min).to_f / [max - min, 1].max) * 100 %>
5
+ <% width_pct = ((value_max - value_min).to_f / [max - min, 1].max) * 100 %>
6
+ <div class="relative w-full h-6" data-controller="hw-range-slider" data-hw-range-slider-min-value="<%= min %>" data-hw-range-slider-max-value="<%= max %>" <%= tag.attributes(local_assigns) %>>
7
+ <div class="absolute top-1/2 -translate-y-1/2 h-2 w-full rounded-full bg-hw-muted">
8
+ <div class="absolute h-full rounded-full bg-hw-primary" style="left: <%= left_pct %>%; width: <%= width_pct %>%" data-hw-range-slider-target="track"></div>
9
+ </div>
10
+ <input type="range" name="<%= name_min %>" value="<%= value_min %>" min="<%= min %>" max="<%= max %>" step="<%= step %>" <%= 'disabled' if disabled %> class="absolute top-0 left-0 w-full h-2 appearance-none bg-transparent pointer-events-none [&::-webkit-slider-thumb]:pointer-events-auto [&::-webkit-slider-thumb]:appearance-none [&::-webkit-slider-thumb]:h-4 [&::-webkit-slider-thumb]:w-4 [&::-webkit-slider-thumb]:rounded-full [&::-webkit-slider-thumb]:bg-hw-primary [&::-webkit-slider-thumb]:cursor-pointer [&::-webkit-slider-thumb]:border-2 [&::-webkit-slider-thumb]:border-hw-background [&::-webkit-slider-thumb]:shadow [&::-moz-range-thumb]:pointer-events-auto [&::-moz-range-thumb]:h-4 [&::-moz-range-thumb]:w-4 [&::-moz-range-thumb]:rounded-full [&::-moz-range-thumb]:bg-hw-primary [&::-moz-range-thumb]:cursor-pointer [&::-moz-range-thumb]:border-2 [&::-moz-range-thumb]:border-hw-background z-10" data-action="input->hw-range-slider#min" data-hw-range-slider-target="min">
11
+ <input type="range" name="<%= name_max %>" value="<%= value_max %>" min="<%= min %>" max="<%= max %>" step="<%= step %>" <%= 'disabled' if disabled %> class="absolute top-0 left-0 w-full h-2 appearance-none bg-transparent pointer-events-none [&::-webkit-slider-thumb]:pointer-events-auto [&::-webkit-slider-thumb]:appearance-none [&::-webkit-slider-thumb]:h-4 [&::-webkit-slider-thumb]:w-4 [&::-webkit-slider-thumb]:rounded-full [&::-webkit-slider-thumb]:bg-hw-primary [&::-webkit-slider-thumb]:cursor-pointer [&::-webkit-slider-thumb]:border-2 [&::-webkit-slider-thumb]:border-hw-background [&::-webkit-slider-thumb]:shadow [&::-moz-range-thumb]:pointer-events-auto [&::-moz-range-thumb]:h-4 [&::-moz-range-thumb]:w-4 [&::-moz-range-thumb]:rounded-full [&::-moz-range-thumb]:bg-hw-primary [&::-moz-range-thumb]:cursor-pointer [&::-moz-range-thumb]:border-2 [&::-moz-range-thumb]:border-hw-background z-20" data-action="input->hw-range-slider#max" data-hw-range-slider-target="max">
12
+ </div>
@@ -0,0 +1,14 @@
1
+ <%# Rating component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/rating', ... %>
3
+
4
+ <div class="flex items-center gap-0.5" data-controller="hw-rating">
5
+ <input type="hidden" name="<%= name %>" value="<%= value %>" data-hw-rating-target="input">
6
+ <% max.times do |i| %>
7
+ <% filled = (i + 1) <= value %>
8
+ <button type="button" class="cursor-pointer <%= filled ? 'text-hw-warning' : 'text-hw-muted-foreground' %>" <%= 'disabled' if readonly %> data-action="<%= 'click->hw-rating#select' unless readonly %>" data-value="<%= i + 1 %>">
9
+ <svg class="<%= SIZES[size] || 'h-5 w-5' %>" fill="<%= filled ? 'currentColor' : 'none' %>" stroke="currentColor" viewBox="0 0 24 24">
10
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="1.5" d="M11.049 2.927c.3-.921 1.603-.921 1.902 0l1.519 4.674a1 1 0 00.95.69h4.915c.969 0 1.371 1.24.588 1.81l-3.976 2.888a1 1 0 00-.363 1.118l1.518 4.674c.3.922-.755 1.688-1.538 1.118l-3.976-2.888a1 1 0 00-1.176 0l-3.976 2.888c-.783.57-1.838-.197-1.538-1.118l1.518-4.674a1 1 0 00-.363-1.118l-3.976-2.888c-.784-.57-.38-1.81.588-1.81h4.914a1 1 0 00.951-.69l1.519-4.674z"/>
11
+ </svg>
12
+ </button>
13
+ <% end %>
14
+ </div>
@@ -0,0 +1,22 @@
1
+ <%# Resizable component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/resizable', ... %>
3
+
4
+ <div data-controller="hw-resizable" class="<%= container_classes %>" <%= tag.attributes(local_assigns) %>>
5
+ <% if direction == :horizontal %>
6
+ <div class="<%= panel_classes %>" data-hw-resizable-target="panel1">
7
+ <%= left %>
8
+ </div>
9
+ <div class="<%= handle_classes %>" data-hw-resizable-target="handle" data-action="pointerdown->hw-resizable#startResize"></div>
10
+ <div class="<%= panel_classes %>" data-hw-resizable-target="panel2">
11
+ <%= right %>
12
+ </div>
13
+ <% else %>
14
+ <div class="<%= panel_classes %>" data-hw-resizable-target="panel1">
15
+ <%= top %>
16
+ </div>
17
+ <div class="<%= handle_classes %>" data-hw-resizable-target="handle" data-action="pointerdown->hw-resizable#startResize"></div>
18
+ <div class="<%= panel_classes %>" data-hw-resizable-target="panel2">
19
+ <%= bottom %>
20
+ </div>
21
+ <% end %>
22
+ </div>
@@ -0,0 +1,39 @@
1
+ <%# Review component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/review', ... %>
3
+
4
+ <article class="border-b border-hw-border pb-6 last:border-0" <%= tag.attributes(local_assigns) %>>
5
+ <div class="flex items-start gap-4">
6
+ <div class="flex h-10 w-10 shrink-0 items-center justify-center rounded-full bg-hw-muted text-sm font-semibold text-hw-muted-foreground">
7
+ <%= initials %>
8
+ </div>
9
+ <div class="min-w-0 flex-1">
10
+ <div class="flex flex-wrap items-center gap-2">
11
+ <span class="text-sm font-semibold text-hw-foreground"><%= author %></span>
12
+ <% if verified %>
13
+ <span class="inline-flex items-center gap-1 rounded-full bg-hw-success/5 px-2 py-0.5 text-xs font-medium text-hw-success ring-1 ring-inset ring-green-600/20">
14
+ <svg class="h-3 w-3" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
15
+ <path fill-rule="evenodd" d="M16.704 4.153a.75.75 0 01.143 1.052l-8 10.5a.75.75 0 01-1.127.075l-4.5-4.5a.75.75 0 011.06-1.06l3.894 3.893 7.48-9.817a.75.75 0 011.05-.143z" clip-rule="evenodd"/>
16
+ </svg>
17
+ Verified
18
+ </span>
19
+ <% end %>
20
+ </div>
21
+ <div class="mt-1 flex items-center gap-2">
22
+ <div class="flex items-center">
23
+ <% 5.times do |i| %>
24
+ <svg class="h-4 w-4 <%= i < rating ? 'text-hw-warning' : 'text-hw-muted-foreground/40' %>" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
25
+ <path fill-rule="evenodd" d="M10.868 2.884c-.321-.772-1.415-.772-1.736 0l-1.83 4.401-4.753.381c-.833.067-1.171 1.107-.536 1.651l3.62 3.102-1.106 4.637c-.194.813.691 1.456 1.405 1.02L10 15.591l4.069 2.485c.713.436 1.598-.207 1.404-1.02l-1.106-4.637 3.62-3.102c.635-.544.297-1.584-.536-1.65l-4.752-.382-1.831-4.401z" clip-rule="evenodd"/>
26
+ </svg>
27
+ <% end %>
28
+ </div>
29
+ <% if date %>
30
+ <time class="text-xs text-hw-muted-foreground"><%= date %></time>
31
+ <% end %>
32
+ </div>
33
+ <% if title %>
34
+ <h4 class="mt-2 text-sm font-semibold text-hw-foreground"><%= title %></h4>
35
+ <% end %>
36
+ <p class="mt-1.5 text-sm leading-relaxed text-hw-muted-foreground"><%= content %></p>
37
+ </div>
38
+ </div>
39
+ </article>
@@ -0,0 +1,58 @@
1
+ <%# RichTextEditor component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/rich_text_editor', ... %>
3
+
4
+ <div class="rounded-lg border border-hw-border bg-hw-background overflow-hidden"
5
+ data-controller="hw-rich-text-editor"
6
+ <%= tag.attributes(local_assigns) %>>
7
+ <div class="flex flex-wrap items-center gap-0.5 border-b border-hw-border bg-hw-muted/50 px-2 py-1.5">
8
+ <% toolbar.each do |btn| %>
9
+ <% case btn %>
10
+ <% when 'bold' %>
11
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#bold" title="Bold">
12
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2.5"><path stroke-linecap="round" stroke-linejoin="round" d="M6 4h8a4 4 0 014 4 4 4 0 01-4 4H6z"/><path stroke-linecap="round" stroke-linejoin="round" d="M6 12h9a4 4 0 014 4 4 4 0 01-4 4H6z"/></svg>
13
+ </button>
14
+ <% when 'italic' %>
15
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#italic" title="Italic">
16
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M10 4h4m-2 0l-4 16m0 0h4"/></svg>
17
+ </button>
18
+ <% when 'underline' %>
19
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#underline" title="Underline">
20
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M7 4v7a5 5 0 0010 0V4M5 21h14"/></svg>
21
+ </button>
22
+ <% when 'strikethrough' %>
23
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#strikethrough" title="Strikethrough">
24
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M16 4H9a3 3 0 00-3 3v0a3 3 0 003 3h6a3 3 0 013 3v0a3 3 0 01-3 3H8M4 12h16"/></svg>
25
+ </button>
26
+ <% when 'link' %>
27
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#link" title="Link">
28
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M13.828 10.172a4 4 0 00-5.656 0l-4 4a4 4 0 105.656 5.656l1.102-1.101m-.758-4.899a4 4 0 005.656 0l4-4a4 4 0 00-5.656-5.656l-1.1 1.1"/></svg>
29
+ </button>
30
+ <% when 'bullet_list' %>
31
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#bulletList" title="Bullet List">
32
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M4 6h16M4 12h16M4 18h16"/></svg>
33
+ </button>
34
+ <% when 'ordered_list' %>
35
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#orderedList" title="Ordered List">
36
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M7 6h13M7 12h13M7 18h13M3 6h.01M3 12h.01M3 18h.01"/></svg>
37
+ </button>
38
+ <% when 'heading' %>
39
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#heading" title="Heading">
40
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M7 4v16M17 4v16M7 12h10"/></svg>
41
+ </button>
42
+ <% when 'quote' %>
43
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#quote" title="Quote">
44
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M8 10h.01M12 10h.01M16 10h.01M9 16H5a2 2 0 01-2-2V6a2 2 0 012-2h14a2 2 0 012 2v8a2 2 0 01-2 2h-5l-5 5v-5z"/></svg>
45
+ </button>
46
+ <% when 'code' %>
47
+ <button type="button" class="h-8 w-8 inline-flex items-center justify-center rounded hover:bg-hw-accent text-hw-muted-foreground hover:text-hw-accent-foreground cursor-pointer" data-action="click->hw-rich-text-editor#code" title="Code">
48
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M10 20l4-16m4 4l4 4-4 4M6 16l-4-4 4-4"/></svg>
49
+ </button>
50
+ <% end %>
51
+ <% end %>
52
+ </div>
53
+ <div class="min-h-[200px] p-4 text-sm text-hw-foreground focus:outline-none"
54
+ contenteditable="true"
55
+ data-hw-rich-text-editor-target="content"
56
+ data-placeholder="<%= placeholder %>"><%= value %></div>
57
+ <input type="hidden" name="<%= name %>" data-hw-rich-text-editor-target="input" value="<%= value %>">
58
+ </div>
@@ -0,0 +1,6 @@
1
+ <%# ScrollArea component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/scroll_area', ... %>
3
+
4
+ <div class="<%= scroll_classes %>" <% if scroll_styles %>style="<%= scroll_styles %>"<% end %> <%= tag.attributes(local_assigns) %>>
5
+ <%= content %>
6
+ </div>
@@ -0,0 +1,59 @@
1
+ <%# Search component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/search', ... %>
3
+
4
+ <div class="<%= container_classes %>"
5
+ data-controller="hw-search"
6
+ data-hw-search-suggestions-value="<%= suggestions.to_json %>"
7
+ <%= tag.attributes(local_assigns) %>>
8
+ <div class="relative">
9
+ <svg class="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-hw-muted-foreground" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
10
+ <circle cx="11" cy="11" r="8"/><path stroke-linecap="round" d="M21 21l-4.35-4.35"/>
11
+ </svg>
12
+ <input type="search"
13
+ name="<%= name %>"
14
+ value="<%= value %>"
15
+ placeholder="<%= placeholder %>"
16
+ class="w-full h-10 rounded-lg border border-hw-border bg-hw-background pl-10 pr-4 text-sm text-hw-foreground placeholder:text-hw-muted-foreground focus:outline-none focus:ring-2 focus:ring-hw-ring focus:border-transparent"
17
+ data-hw-search-target="input"
18
+ data-action="input->hw-search#debounce focus->hw-search#showSuggestions blur->hw-search#hideSuggestions">
19
+ <% if with_filters %>
20
+ <button type="button"
21
+ class="absolute right-2 top-1/2 -translate-y-1/2 h-7 w-7 rounded-md flex items-center justify-center hover:bg-hw-muted transition-colors"
22
+ data-action="click->hw-search#toggleFilters">
23
+ <svg class="h-4 w-4 text-hw-muted-foreground" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
24
+ <path stroke-linecap="round" stroke-linejoin="round" d="M3 4a1 1 0 011-1h16a1 1 0 011 1v2.586a1 1 0 01-.293.707l-6.414 6.414a1 1 0 00-.293.707V17l-4 4v-6.586a1 1 0 00-.293-.707L3.293 7.293A1 1 0 013 6.586V4z"/>
25
+ </svg>
26
+ <span class="sr-only">Filters</span>
27
+ </button>
28
+ <% end %>
29
+ </div>
30
+ <div class="hidden absolute top-full left-0 right-0 mt-1 rounded-lg border border-hw-border bg-hw-popover shadow-lg z-50"
31
+ data-hw-search-target="suggestions">
32
+ <ul class="py-1 max-h-60 overflow-auto" data-hw-search-target="list">
33
+ <% suggestions.each do |suggestion| %>
34
+ <li>
35
+ <button type="button"
36
+ class="w-full px-3 py-2 text-left text-sm hover:bg-hw-accent hover:text-hw-accent-foreground transition-colors"
37
+ data-action="click->hw-search#select"
38
+ data-value="<%= suggestion.is_a?(Hash) ? suggestion[:value] : suggestion %>">
39
+ <% if suggestion.is_a?(Hash) %>
40
+ <span class="font-medium"><%= suggestion[:label] %></span>
41
+ <% if suggestion[:description] %>
42
+ <span class="ml-2 text-xs text-hw-muted-foreground"><%= suggestion[:description] %></span>
43
+ <% end %>
44
+ <% else %>
45
+ <%= suggestion %>
46
+ <% end %>
47
+ </button>
48
+ </li>
49
+ <% end %>
50
+ </ul>
51
+ </div>
52
+ <% if with_filters %>
53
+ <div class="hidden mt-2 rounded-lg border border-hw-border bg-hw-card p-3" data-hw-search-target="filters">
54
+ <p class="text-xs font-medium text-hw-muted-foreground mb-2">Filters</p>
55
+ <div class="flex flex-wrap gap-2" data-hw-search-target="filterContainer">
56
+ </div>
57
+ </div>
58
+ <% end %>
59
+ </div>
@@ -0,0 +1,9 @@
1
+ <%# SearchInput component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/search_input', ... %>
3
+
4
+ <div class="relative">
5
+ <div class="absolute left-3 top-1/2 -translate-y-1/2 text-hw-muted-foreground">
6
+ <svg class="h-4 w-4" fill="none" stroke="currentColor" stroke-width="2" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"/></svg>
7
+ </div>
8
+ <input type="search" name="<%= name %>" value="<%= value %>" placeholder="<%= placeholder %>" <%= 'disabled' if disabled %> class="<%= input_classes %>" <%= tag.attributes(local_assigns) %>>
9
+ </div>
@@ -0,0 +1,9 @@
1
+ <%# SectionHeading component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/section_heading', ... %>
3
+
4
+ <div class="flex flex-col gap-1 <%= container_classes %>" <%= tag.attributes(local_assigns) %>>
5
+ <h2 class="text-xl font-semibold text-hw-foreground"><%= title %></h2>
6
+ <% if description %>
7
+ <p class="text-sm text-hw-muted-foreground"><%= description %></p>
8
+ <% end %>
9
+ </div>
@@ -0,0 +1,30 @@
1
+ <%# Select component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/select', name: "country", options: [["US","us"]], placeholder: "Choose" %>
3
+
4
+ <%
5
+ name = local_assigns.fetch(:name, "")
6
+ options = local_assigns.fetch(:options, [])
7
+ placeholder = local_assigns.fetch(:placeholder, "Select...")
8
+ selected = local_assigns[:selected]
9
+ disabled = local_assigns.fetch(:disabled, false)
10
+ multiple = local_assigns.fetch(:multiple, false)
11
+
12
+ select_cls = "flex h-10 w-full rounded-md border border-hw-input bg-hw-background px-3 py-2 text-sm ring-offset-hw-background focus:outline-none focus:ring-2 focus:ring-hw-ring focus:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50"
13
+ %>
14
+
15
+ <select name="<%= name %>" <%= 'disabled' if disabled %> <%= 'multiple' if multiple %> class="<%= select_cls %>">
16
+ <option value="" disabled <%= 'selected' unless selected %>><%= placeholder %></option>
17
+ <% options.each do |opt| %>
18
+ <% if opt.is_a?(Array) %>
19
+ <option value="<%= opt.last %>" <%= 'selected' if Array(selected).include?(opt.last) %>><%= opt.first %></option>
20
+ <% elsif opt.is_a?(Hash) %>
21
+ <optgroup label="<%= opt[:label] %>">
22
+ <% opt[:options].each do |o| %>
23
+ <option value="<%= o.last %>" <%= 'selected' if Array(selected).include?(o.last) %>><%= o.first %></option>
24
+ <% end %>
25
+ </optgroup>
26
+ <% else %>
27
+ <option value="<%= opt %>" <%= 'selected' if Array(selected).include?(opt) %>><%= opt %></option>
28
+ <% end %>
29
+ <% end %>
30
+ </select>
@@ -0,0 +1,12 @@
1
+ <%# SemiCircleProgress component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/semi_circle_progress', ... %>
3
+
4
+ <div class="relative inline-flex items-center justify-center" <%= tag.attributes(local_assigns) %>>
5
+ <svg width="<%= size %>" height="<%= size / 2 + 8 %>" viewBox="0 0 <%= size %> <%= size / 2 + 8 %>">
6
+ <path d="M <%= center_x - radius %> <%= center_y %> A <%= radius %> <%= radius %> 0 0 1 <%= center_x + radius %> <%= center_y %>" fill="none" stroke-width="8" class="stroke-hw-muted" stroke-linecap="round"/>
7
+ <path d="M <%= center_x - radius %> <%= center_y %> A <%= radius %> <%= radius %> 0 0 1 <%= center_x + radius %> <%= center_y %>" fill="none" stroke-width="8" class="<%= color_class %> transition-all duration-300" stroke-linecap="round" stroke-dasharray="<%= circumference %>" stroke-dashoffset="<%= stroke_dashoffset %>"/>
8
+ </svg>
9
+ <% if show_label %>
10
+ <span class="absolute bottom-0 text-sm font-medium"><%= percentage %>%</span>
11
+ <% end %>
12
+ </div>
@@ -0,0 +1,18 @@
1
+ <%# Separator component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/separator', orientation: :horizontal %>
3
+
4
+ <%
5
+ orientation = local_assigns.fetch(:orientation, :horizontal)
6
+ label = local_assigns[:label]
7
+ decorative = local_assigns.fetch(:decorative, true)
8
+ %>
9
+
10
+ <% if label %>
11
+ <div class="relative flex items-center" role="none">
12
+ <div class="shrink-0 bg-hw-border h-[1px] w-full"></div>
13
+ <span class="px-2 text-xs text-hw-muted-foreground font-medium"><%= label %></span>
14
+ <div class="shrink-0 bg-hw-border h-[1px] w-full"></div>
15
+ </div>
16
+ <% else %>
17
+ <div class="<%= orientation == :vertical ? 'shrink-0 bg-hw-border h-full w-[1px]' : 'shrink-0 bg-hw-border h-[1px] w-full' %>" role="<%= decorative ? 'none' : 'separator' %>"></div>
18
+ <% end %>
@@ -0,0 +1,45 @@
1
+ <%# Sheet component — ERB partial %>
2
+ <%# Usage: render 'hotwirebits/sheet', ... %>
3
+
4
+ <div data-controller="hw-dialog"
5
+ data-hw-dialog-open-value="<%= open %>">
6
+ <div class="fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0"
7
+ data-state="<%= open ? 'open' : 'closed' %>"
8
+ data-hw-dialog-target="overlay"
9
+ data-action="click->hw-dialog#close"
10
+ aria-hidden="true"></div>
11
+ <div class="fixed z-50 gap-4 bg-hw-background p-6 shadow-lg transition ease-in-out data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:duration-300 data-[state=open]:duration-500 <%= side_config[:panel] %>"
12
+ data-state="<%= open ? 'open' : 'closed' %>"
13
+ data-hw-dialog-target="panel"
14
+ role="dialog"
15
+ aria-modal="true"
16
+ <%= tag.attributes(local_assigns) %>>
17
+ <div class="flex flex-col space-y-2">
18
+ <% if header? %>
19
+ <%= header %>
20
+ <% elsif title %>
21
+ <div class="flex items-center justify-between">
22
+ <h2 class="text-lg font-semibold tracking-tight"><%= title %></h2>
23
+ <% if closable %>
24
+ <button type="button"
25
+ class="rounded-sm opacity-70 ring-offset-hw-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-hw-ring focus:ring-offset-2"
26
+ data-action="click->hw-dialog#close">
27
+ <svg class="h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
28
+ <path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12"/>
29
+ </svg>
30
+ <span class="sr-only">Close</span>
31
+ </button>
32
+ <% end %>
33
+ </div>
34
+ <% end %>
35
+ </div>
36
+ <div class="py-4 text-sm">
37
+ <%= content %>
38
+ </div>
39
+ <% if footer? %>
40
+ <div class="flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2">
41
+ <%= footer %>
42
+ </div>
43
+ <% end %>
44
+ </div>
45
+ </div>