hakumi_components 0.1.17.pre → 0.1.19.pre

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 (647) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +33 -3
  3. data/README.md +218 -369
  4. data/app/assets/javascripts/hakumi_components.js +40 -34
  5. data/app/assets/stylesheets/hakumi_components.css +1 -1
  6. data/app/components/{hakumi → hakumi_components}/admin_panel/component.html.erb +2 -2
  7. data/app/components/hakumi_components/admin_panel/component.rb +50 -0
  8. data/app/components/hakumi_components/affix/component.rb +112 -0
  9. data/app/components/{hakumi → hakumi_components}/alert/component.html.erb +3 -3
  10. data/app/components/hakumi_components/alert/component.rb +134 -0
  11. data/app/components/hakumi_components/anchor/component.rb +88 -0
  12. data/app/components/{hakumi → hakumi_components}/anchor/link/component.html.erb +1 -1
  13. data/app/components/hakumi_components/anchor/link/component.rb +53 -0
  14. data/app/components/hakumi_components/autocomplete/component.html.erb +94 -0
  15. data/app/components/hakumi_components/autocomplete/component.rb +228 -0
  16. data/app/components/hakumi_components/avatar/component.rb +124 -0
  17. data/app/components/hakumi_components/badge/component.rb +139 -0
  18. data/app/components/hakumi_components/base_component.rb +391 -0
  19. data/app/components/hakumi_components/breadcrumb/component.rb +38 -0
  20. data/app/components/hakumi_components/breadcrumb/item/component.html.erb +17 -0
  21. data/app/components/hakumi_components/breadcrumb/item/component.rb +83 -0
  22. data/app/components/hakumi_components/breadcrumb/item/menu_entry.rb +71 -0
  23. data/app/components/hakumi_components/button/component.rb +163 -0
  24. data/app/components/{hakumi → hakumi_components}/calendar/component.html.erb +24 -24
  25. data/app/components/hakumi_components/calendar/component.rb +471 -0
  26. data/app/components/hakumi_components/calendar/day_cell.rb +62 -0
  27. data/app/components/hakumi_components/calendar/display_model.rb +237 -0
  28. data/app/components/hakumi_components/calendar/event_entry.rb +54 -0
  29. data/app/components/hakumi_components/calendar/locale_pack.rb +28 -0
  30. data/app/components/hakumi_components/calendar/panel_entry.rb +48 -0
  31. data/app/components/hakumi_components/calendar/week_row.rb +28 -0
  32. data/app/components/{hakumi → hakumi_components}/card/component.html.erb +2 -2
  33. data/app/components/hakumi_components/card/component.rb +121 -0
  34. data/app/components/hakumi_components/card/grid/component.html.erb +3 -0
  35. data/app/components/hakumi_components/card/grid/component.rb +43 -0
  36. data/app/components/hakumi_components/card/meta/component.html.erb +23 -0
  37. data/app/components/hakumi_components/card/meta/component.rb +64 -0
  38. data/app/components/hakumi_components/carousel/autoplay_config.rb +27 -0
  39. data/app/components/{hakumi → hakumi_components}/carousel/component.html.erb +2 -2
  40. data/app/components/hakumi_components/carousel/component.rb +165 -0
  41. data/app/components/hakumi_components/carousel/dots_config.rb +36 -0
  42. data/app/components/hakumi_components/cascader/component.html.erb +59 -0
  43. data/app/components/hakumi_components/cascader/component.rb +168 -0
  44. data/app/components/{hakumi → hakumi_components}/checkbox/component.html.erb +5 -5
  45. data/app/components/hakumi_components/checkbox/component.rb +98 -0
  46. data/app/components/hakumi_components/checkbox/group/component.html.erb +12 -0
  47. data/app/components/hakumi_components/checkbox/group/component.rb +91 -0
  48. data/app/components/hakumi_components/checkbox/group/option.rb +34 -0
  49. data/app/components/{hakumi → hakumi_components}/collapse/component.html.erb +1 -1
  50. data/app/components/hakumi_components/collapse/component.rb +133 -0
  51. data/app/components/hakumi_components/collapse/panel/component.rb +108 -0
  52. data/app/components/{hakumi → hakumi_components}/color_picker/component.html.erb +21 -22
  53. data/app/components/hakumi_components/color_picker/component.rb +279 -0
  54. data/app/components/hakumi_components/color_picker/preset_group.rb +27 -0
  55. data/app/components/hakumi_components/concerns/form_field.rb +236 -0
  56. data/app/components/hakumi_components/concerns/form_field_contract.rb +57 -0
  57. data/app/components/hakumi_components/concerns/form_field_interface.rb +16 -0
  58. data/app/components/hakumi_components/concerns/input_control.rb +111 -0
  59. data/app/components/hakumi_components/concerns/input_control_contract.rb +47 -0
  60. data/app/components/hakumi_components/concerns/input_control_interface.rb +34 -0
  61. data/app/components/hakumi_components/concerns/selection_control.rb +122 -0
  62. data/app/components/hakumi_components/concerns/selection_control_contract.rb +47 -0
  63. data/app/components/hakumi_components/concerns/selection_control_interface.rb +34 -0
  64. data/app/components/{hakumi → hakumi_components}/container/component.rb +31 -16
  65. data/app/components/{hakumi → hakumi_components}/date_picker/component.html.erb +4 -6
  66. data/app/components/hakumi_components/date_picker/component.rb +254 -0
  67. data/app/components/{hakumi → hakumi_components}/date_picker/range_picker.html.erb +4 -4
  68. data/app/components/hakumi_components/date_picker/range_picker.rb +267 -0
  69. data/app/components/hakumi_components/date_picker/shared_rendering.rb +289 -0
  70. data/app/components/{hakumi → hakumi_components}/descriptions/component.html.erb +6 -6
  71. data/app/components/hakumi_components/descriptions/component.rb +219 -0
  72. data/app/components/hakumi_components/descriptions/item/component.rb +52 -0
  73. data/app/components/hakumi_components/divider/component.rb +86 -0
  74. data/app/components/{hakumi → hakumi_components}/drawer/component.html.erb +3 -3
  75. data/app/components/hakumi_components/drawer/component.rb +191 -0
  76. data/app/components/hakumi_components/dropdown/component.rb +94 -0
  77. data/app/components/{hakumi → hakumi_components}/dropdown/divider/component.rb +6 -2
  78. data/app/components/hakumi_components/dropdown/item/component.rb +73 -0
  79. data/app/components/{hakumi → hakumi_components}/empty/component.rb +68 -33
  80. data/app/components/hakumi_components/flex/component.rb +96 -0
  81. data/app/components/hakumi_components/float_button/back_top/component.rb +86 -0
  82. data/app/components/hakumi_components/float_button/component.rb +289 -0
  83. data/app/components/hakumi_components/float_button/group/component.rb +214 -0
  84. data/app/components/hakumi_components/float_button/group_cluster/component.rb +133 -0
  85. data/app/components/hakumi_components/float_button/group_spec.rb +259 -0
  86. data/app/components/hakumi_components/float_button/item_spec.rb +115 -0
  87. data/app/components/hakumi_components/float_button/position_config.rb +68 -0
  88. data/app/components/{hakumi → hakumi_components}/form/item/component.rb +36 -29
  89. data/app/components/hakumi_components/grid/col/component.rb +169 -0
  90. data/app/components/hakumi_components/grid/row/component.rb +97 -0
  91. data/app/components/{hakumi → hakumi_components}/icon/component.rb +53 -53
  92. data/app/components/{hakumi → hakumi_components}/image/component.html.erb +17 -17
  93. data/app/components/hakumi_components/image/component.rb +175 -0
  94. data/app/components/{hakumi → hakumi_components}/image/preview_group/component.html.erb +10 -10
  95. data/app/components/hakumi_components/image/preview_group/component.rb +75 -0
  96. data/app/components/hakumi_components/image/preview_group/item.rb +48 -0
  97. data/app/components/{hakumi → hakumi_components}/input/component.html.erb +1 -1
  98. data/app/components/hakumi_components/input/component.rb +261 -0
  99. data/app/components/hakumi_components/input/password/component.rb +84 -0
  100. data/app/components/{hakumi → hakumi_components}/input/text_area/component.html.erb +2 -2
  101. data/app/components/hakumi_components/input/text_area/component.rb +178 -0
  102. data/app/components/{hakumi → hakumi_components}/input_number/component.html.erb +9 -9
  103. data/app/components/hakumi_components/input_number/component.rb +280 -0
  104. data/app/components/hakumi_components/layout/component.html.erb +1 -0
  105. data/app/components/hakumi_components/layout/component.rb +28 -0
  106. data/app/components/hakumi_components/layout/content/component.html.erb +1 -0
  107. data/app/components/hakumi_components/layout/content/component.rb +29 -0
  108. data/app/components/hakumi_components/layout/footer/component.html.erb +1 -0
  109. data/app/components/hakumi_components/layout/footer/component.rb +29 -0
  110. data/app/components/hakumi_components/layout/header/component.html.erb +1 -0
  111. data/app/components/hakumi_components/layout/header/component.rb +29 -0
  112. data/app/components/hakumi_components/layout/sider/component.html.erb +1 -0
  113. data/app/components/hakumi_components/layout/sider/component.rb +90 -0
  114. data/app/components/hakumi_components/mentions/coercion.rb +46 -0
  115. data/app/components/{hakumi → hakumi_components}/mentions/component.html.erb +6 -6
  116. data/app/components/hakumi_components/mentions/component.rb +229 -0
  117. data/app/components/hakumi_components/mentions/option.rb +30 -0
  118. data/app/components/hakumi_components/menu/component.rb +117 -0
  119. data/app/components/{hakumi → hakumi_components}/menu/divider/component.rb +6 -2
  120. data/app/components/hakumi_components/menu/group/component.rb +48 -0
  121. data/app/components/hakumi_components/menu/item/component.rb +70 -0
  122. data/app/components/hakumi_components/menu/sub_menu/component.rb +104 -0
  123. data/app/components/hakumi_components/message/component.rb +141 -0
  124. data/app/components/{hakumi → hakumi_components}/modal/component.html.erb +7 -7
  125. data/app/components/hakumi_components/modal/component.rb +152 -0
  126. data/app/components/hakumi_components/modal/confirm/component.html.erb +23 -0
  127. data/app/components/hakumi_components/modal/confirm/component.rb +87 -0
  128. data/app/components/hakumi_components/modal/error/component.rb +20 -0
  129. data/app/components/hakumi_components/modal/info/component.rb +17 -0
  130. data/app/components/hakumi_components/modal/status_component.rb +150 -0
  131. data/app/components/hakumi_components/modal/success/component.rb +17 -0
  132. data/app/components/hakumi_components/modal/warning/component.rb +26 -0
  133. data/app/components/hakumi_components/notification/component.rb +160 -0
  134. data/app/components/{hakumi → hakumi_components}/pagination/component.html.erb +8 -12
  135. data/app/components/hakumi_components/pagination/component.rb +238 -0
  136. data/app/components/hakumi_components/pagination/page_item.rb +49 -0
  137. data/app/components/{hakumi → hakumi_components}/popconfirm/component.html.erb +2 -2
  138. data/app/components/hakumi_components/popconfirm/component.rb +120 -0
  139. data/app/components/{hakumi → hakumi_components}/popover/component.html.erb +2 -2
  140. data/app/components/hakumi_components/popover/component.rb +196 -0
  141. data/app/components/hakumi_components/progress/attribute_renderer.rb +354 -0
  142. data/app/components/hakumi_components/progress/circle_geometry.rb +76 -0
  143. data/app/components/{hakumi → hakumi_components}/progress/component.html.erb +19 -18
  144. data/app/components/hakumi_components/progress/component.rb +393 -0
  145. data/app/components/hakumi_components/progress/controller_locals_parser.rb +130 -0
  146. data/app/components/hakumi_components/progress/info_tooltip_policy.rb +39 -0
  147. data/app/components/hakumi_components/progress/info_value.rb +75 -0
  148. data/app/components/hakumi_components/progress/status_state.rb +31 -0
  149. data/app/components/hakumi_components/progress/steps_renderer.rb +149 -0
  150. data/app/components/hakumi_components/progress/stroke_gradient_value.rb +90 -0
  151. data/app/components/hakumi_components/qr_code/component.rb +405 -0
  152. data/app/components/{hakumi → hakumi_components}/radio/component.html.erb +3 -3
  153. data/app/components/hakumi_components/radio/component.rb +113 -0
  154. data/app/components/{hakumi → hakumi_components}/radio/group/component.html.erb +10 -10
  155. data/app/components/hakumi_components/radio/group/component.rb +181 -0
  156. data/app/components/hakumi_components/radio/group/option.rb +43 -0
  157. data/app/components/{hakumi → hakumi_components}/rate/component.html.erb +2 -2
  158. data/app/components/{hakumi → hakumi_components}/rate/component.rb +58 -12
  159. data/app/components/hakumi_components/result/component.rb +175 -0
  160. data/app/components/{hakumi → hakumi_components}/segmented/component.html.erb +1 -1
  161. data/app/components/hakumi_components/segmented/component.rb +232 -0
  162. data/app/components/hakumi_components/segmented/option.rb +191 -0
  163. data/app/components/{hakumi → hakumi_components}/select/component.html.erb +22 -22
  164. data/app/components/hakumi_components/select/component.rb +176 -0
  165. data/app/components/hakumi_components/selection_control/coercion.rb +161 -0
  166. data/app/components/hakumi_components/selection_control/entry.rb +22 -0
  167. data/app/components/hakumi_components/selection_control/option.rb +42 -0
  168. data/app/components/hakumi_components/selection_control/option_group.rb +34 -0
  169. data/app/components/hakumi_components/selection_control/tree_node.rb +123 -0
  170. data/app/components/hakumi_components/skeleton/avatar/component.rb +88 -0
  171. data/app/components/hakumi_components/skeleton/avatar_config.rb +79 -0
  172. data/app/components/hakumi_components/skeleton/button/component.rb +77 -0
  173. data/app/components/hakumi_components/skeleton/component.rb +186 -0
  174. data/app/components/{hakumi → hakumi_components}/skeleton/image/component.html.erb +1 -1
  175. data/app/components/hakumi_components/skeleton/image/component.rb +104 -0
  176. data/app/components/hakumi_components/skeleton/input/component.rb +66 -0
  177. data/app/components/hakumi_components/skeleton/node/component.rb +59 -0
  178. data/app/components/hakumi_components/skeleton/paragraph_config.rb +92 -0
  179. data/app/components/hakumi_components/skeleton/title_config.rb +31 -0
  180. data/app/components/{hakumi → hakumi_components}/slider/component.html.erb +11 -11
  181. data/app/components/hakumi_components/slider/component.rb +347 -0
  182. data/app/components/hakumi_components/slider/mark.rb +27 -0
  183. data/app/components/hakumi_components/space/compact/component.html.erb +1 -0
  184. data/app/components/hakumi_components/space/compact/component.rb +52 -0
  185. data/app/components/{hakumi → hakumi_components}/space/component.html.erb +1 -1
  186. data/app/components/hakumi_components/space/component.rb +109 -0
  187. data/app/components/hakumi_components/spin/component.rb +270 -0
  188. data/app/components/hakumi_components/splitter/component.html.erb +9 -0
  189. data/app/components/hakumi_components/splitter/component.rb +71 -0
  190. data/app/components/hakumi_components/splitter/panel/component.html.erb +1 -0
  191. data/app/components/hakumi_components/splitter/panel/component.rb +98 -0
  192. data/app/components/hakumi_components/statistic/component.rb +371 -0
  193. data/app/components/{hakumi → hakumi_components}/steps/component.html.erb +4 -4
  194. data/app/components/hakumi_components/steps/component.rb +177 -0
  195. data/app/components/hakumi_components/steps/item/component.rb +130 -0
  196. data/app/components/{hakumi → hakumi_components}/switch/component.html.erb +2 -3
  197. data/app/components/hakumi_components/switch/component.rb +150 -0
  198. data/app/components/hakumi_components/table/column/component.rb +137 -0
  199. data/app/components/hakumi_components/table/column_definition.rb +181 -0
  200. data/app/components/{hakumi → hakumi_components}/table/column_group/component.rb +35 -24
  201. data/app/components/{hakumi → hakumi_components}/table/component.html.erb +22 -31
  202. data/app/components/hakumi_components/table/component.rb +470 -0
  203. data/app/components/hakumi_components/table/concerns/attribute_helpers.rb +22 -0
  204. data/app/components/hakumi_components/table/concerns/cell_rendering.rb +277 -0
  205. data/app/components/hakumi_components/table/concerns/columns.rb +421 -0
  206. data/app/components/hakumi_components/table/concerns/editable.rb +68 -0
  207. data/app/components/hakumi_components/table/concerns/ellipsis.rb +64 -0
  208. data/app/components/hakumi_components/table/concerns/fixed_columns.rb +94 -0
  209. data/app/components/hakumi_components/table/concerns/surface_rendering.rb +365 -0
  210. data/app/components/hakumi_components/table/configs.rb +518 -0
  211. data/app/components/hakumi_components/table/definition_types.rb +40 -0
  212. data/app/components/hakumi_components/table/display_rows.rb +56 -0
  213. data/app/components/hakumi_components/table/ellipsis_config.rb +143 -0
  214. data/app/components/hakumi_components/table/fixed_offset.rb +36 -0
  215. data/app/components/hakumi_components/table/header_cell.rb +23 -0
  216. data/app/components/hakumi_components/table/row_record.rb +60 -0
  217. data/app/components/hakumi_components/table/row_render_state.rb +85 -0
  218. data/app/components/{hakumi → hakumi_components}/tabs/component.html.erb +11 -11
  219. data/app/components/hakumi_components/tabs/component.rb +246 -0
  220. data/app/components/hakumi_components/tabs/indicator_config.rb +34 -0
  221. data/app/components/hakumi_components/tabs/item/component.rb +147 -0
  222. data/app/components/hakumi_components/tag/component.rb +282 -0
  223. data/app/components/{hakumi → hakumi_components}/tag/group/component.rb +20 -8
  224. data/app/components/{hakumi → hakumi_components}/time_picker/component.html.erb +10 -10
  225. data/app/components/{hakumi → hakumi_components}/time_picker/component.rb +75 -21
  226. data/app/components/hakumi_components/timeline/component.html.erb +8 -0
  227. data/app/components/hakumi_components/timeline/component.rb +301 -0
  228. data/app/components/{hakumi → hakumi_components}/timeline/item/component.html.erb +2 -2
  229. data/app/components/hakumi_components/timeline/item/component.rb +198 -0
  230. data/app/components/{hakumi → hakumi_components}/tooltip/component.html.erb +1 -1
  231. data/app/components/hakumi_components/tooltip/component.rb +146 -0
  232. data/app/components/hakumi_components/tour/component.rb +183 -0
  233. data/app/components/hakumi_components/tour/step.rb +86 -0
  234. data/app/components/hakumi_components/transfer/coercion.rb +66 -0
  235. data/app/components/hakumi_components/transfer/component.html.erb +204 -0
  236. data/app/components/hakumi_components/transfer/component.rb +395 -0
  237. data/app/components/hakumi_components/transfer/item.rb +54 -0
  238. data/app/components/hakumi_components/transfer/operations.rb +31 -0
  239. data/app/components/{hakumi → hakumi_components}/tree/component.html.erb +1 -1
  240. data/app/components/hakumi_components/tree/component.rb +342 -0
  241. data/app/components/hakumi_components/tree/concerns/node_rendering.rb +365 -0
  242. data/app/components/hakumi_components/tree/node.rb +276 -0
  243. data/app/components/{hakumi → hakumi_components}/tree_select/component.html.erb +27 -13
  244. data/app/components/hakumi_components/tree_select/component.rb +270 -0
  245. data/app/components/hakumi_components/typography/base_component.rb +142 -0
  246. data/app/components/hakumi_components/typography/link/component.rb +91 -0
  247. data/app/components/{hakumi → hakumi_components}/typography/paragraph/component.rb +7 -1
  248. data/{sig/hakumi/typography/text/component.rbs → app/components/hakumi_components/typography/text/component.rb} +3 -1
  249. data/app/components/hakumi_components/typography/title/component.rb +84 -0
  250. data/app/components/hakumi_components/upload/component.html.erb +135 -0
  251. data/app/components/hakumi_components/upload/component.rb +367 -0
  252. data/app/components/hakumi_components/upload/file_entry.rb +53 -0
  253. data/app/controllers/{hakumi → hakumi_components}/components_controller.rb +3 -2
  254. data/app/form_builders/hakumi_components/form_builder/contracts.rb +55 -0
  255. data/app/form_builders/hakumi_components/form_builder/field_context.rb +43 -0
  256. data/app/form_builders/hakumi_components/form_builder/model_binding.rb +163 -0
  257. data/app/form_builders/hakumi_components/form_builder.rb +403 -0
  258. data/app/helpers/hakumi_components/form_helper.rb +43 -0
  259. data/app/javascript/hakumi_components/controllers/hakumi/admin_panel_controller.js +5 -7
  260. data/app/javascript/hakumi_components/controllers/hakumi/back_top_controller.js +1 -1
  261. data/app/javascript/hakumi_components/controllers/hakumi/button_controller.js +108 -2
  262. data/app/javascript/hakumi_components/controllers/hakumi/calendar_controller.js +183 -95
  263. data/app/javascript/hakumi_components/controllers/hakumi/color_picker_controller.js +23 -285
  264. data/app/javascript/hakumi_components/controllers/hakumi/date_picker_controller.js +274 -262
  265. data/app/javascript/hakumi_components/controllers/hakumi/float_button_group_controller.js +2 -2
  266. data/app/javascript/hakumi_components/controllers/hakumi/message_controller.js +4 -2
  267. data/app/javascript/hakumi_components/controllers/hakumi/modal_controller.js +119 -125
  268. data/app/javascript/hakumi_components/controllers/hakumi/table/editable.js +291 -0
  269. data/app/javascript/hakumi_components/controllers/hakumi/table_controller.js +166 -366
  270. data/app/javascript/hakumi_components/controllers/hakumi/tabs_controller.js +8 -2
  271. data/app/javascript/hakumi_components/controllers/hakumi/tag_controller.js +27 -25
  272. data/app/javascript/hakumi_components/controllers/hakumi/tag_group_controller.js +19 -18
  273. data/app/javascript/hakumi_components/controllers/hakumi/theme_controller.js +116 -117
  274. data/app/javascript/hakumi_components/controllers/hakumi/transfer_controller.js +17 -1
  275. data/app/javascript/hakumi_components/controllers/hakumi/tree_controller.js +363 -78
  276. data/app/javascript/hakumi_components/controllers/hakumi/typography_controller.js +3 -3
  277. data/app/javascript/hakumi_components/controllers/hakumi/upload_controller.js +320 -204
  278. data/app/javascript/hakumi_components/core/render_component.js +37 -11
  279. data/app/javascript/hakumi_components/utils/color_helper.js +262 -0
  280. data/app/javascript/stylesheets/_base.scss +9 -0
  281. data/app/javascript/stylesheets/_hakumi_components.scss +1 -0
  282. data/app/javascript/stylesheets/components/_breadcrumb.scss +2 -2
  283. data/app/javascript/stylesheets/components/_calendar.scss +13 -13
  284. data/app/javascript/stylesheets/components/_cascader.scss +5 -5
  285. data/app/javascript/stylesheets/components/_checkbox.scss +9 -11
  286. data/app/javascript/stylesheets/components/_color_picker.scss +11 -11
  287. data/app/javascript/stylesheets/components/_date_picker.scss +4 -4
  288. data/app/javascript/stylesheets/components/_descriptions.scss +2 -2
  289. data/app/javascript/stylesheets/components/_drawer.scss +3 -3
  290. data/app/javascript/stylesheets/components/_dropdown.scss +2 -2
  291. data/app/javascript/stylesheets/components/_flex.scss +1 -1
  292. data/app/javascript/stylesheets/components/_float_button.scss +5 -5
  293. data/app/javascript/stylesheets/components/_form_item.scss +92 -0
  294. data/app/javascript/stylesheets/components/_image.scss +15 -15
  295. data/app/javascript/stylesheets/components/_input.scss +23 -113
  296. data/app/javascript/stylesheets/components/_layout.scss +27 -26
  297. data/app/javascript/stylesheets/components/_menu.scss +15 -15
  298. data/app/javascript/stylesheets/components/_modal.scss +13 -13
  299. data/app/javascript/stylesheets/components/_notification.scss +3 -3
  300. data/app/javascript/stylesheets/components/_popover.scss +1 -1
  301. data/app/javascript/stylesheets/components/_segmented.scss +3 -3
  302. data/app/javascript/stylesheets/components/_select.scss +6 -6
  303. data/app/javascript/stylesheets/components/_slider.scss +1 -1
  304. data/app/javascript/stylesheets/components/_spin.scss +2 -2
  305. data/app/javascript/stylesheets/components/_steps.scss +10 -10
  306. data/app/javascript/stylesheets/components/_switch.scss +11 -10
  307. data/app/javascript/stylesheets/components/_table.scss +6 -6
  308. data/app/javascript/stylesheets/components/_tag.scss +2 -2
  309. data/app/javascript/stylesheets/components/_tooltip.scss +4 -4
  310. data/app/javascript/stylesheets/components/_tree_select.scss +3 -3
  311. data/app/javascript/stylesheets/components/_typography.scss +3 -3
  312. data/app/javascript/stylesheets/components/_upload.scss +4 -4
  313. data/app/services/hakumi_components/component_handler.rb +124 -0
  314. data/app/services/hakumi_components/icon/loader.rb +112 -0
  315. data/app/services/hakumi_components/illustrations/loader.rb +51 -0
  316. data/app/views/hakumi/_admin_panel.html.erb +1 -1
  317. data/app/views/hakumi/_affix.html.erb +1 -1
  318. data/app/views/hakumi/_alert.html.erb +4 -1
  319. data/app/views/hakumi/_confirm.html.erb +1 -1
  320. data/app/views/hakumi/_drawer.html.erb +2 -2
  321. data/app/views/hakumi/_message.html.erb +1 -1
  322. data/app/views/hakumi/_modal.html.erb +1 -1
  323. data/app/views/hakumi/_notification.html.erb +1 -1
  324. data/app/views/hakumi/_popconfirm.html.erb +1 -1
  325. data/app/views/hakumi/_popover.html.erb +1 -1
  326. data/app/views/hakumi/_qr_code.html.erb +1 -1
  327. data/app/views/hakumi/_result.html.erb +1 -1
  328. data/app/views/hakumi/_segmented.html.erb +1 -1
  329. data/app/views/hakumi/_skeleton.html.erb +1 -1
  330. data/app/views/hakumi/_spin.html.erb +1 -1
  331. data/app/views/hakumi/_statistic.html.erb +12 -12
  332. data/app/views/hakumi/_table.html.erb +1 -1
  333. data/app/views/hakumi/_tag.html.erb +1 -1
  334. data/app/views/hakumi/_timeline.html.erb +1 -1
  335. data/app/views/hakumi/_tree.html.erb +1 -1
  336. data/lib/generators/{hakumi → hakumi_components}/install_generator.rb +17 -4
  337. data/lib/hakumi_components/documentation/models.rb +290 -0
  338. data/lib/hakumi_components/documentation/node.rb +73 -0
  339. data/lib/hakumi_components/documentation.rb +141 -64
  340. data/lib/hakumi_components/engine.rb +26 -14
  341. data/lib/hakumi_components/file_size_checker.rb +215 -0
  342. data/lib/hakumi_components/locales/en.yml +115 -0
  343. data/lib/hakumi_components/rails/attribute_introspection.rb +40 -108
  344. data/lib/hakumi_components/rails/model_reflection.rb +154 -0
  345. data/lib/hakumi_components/rails/validation_introspection.rb +26 -74
  346. data/lib/hakumi_components/rails/validation_mapper.rb +149 -202
  347. data/lib/hakumi_components/rails.rb +3 -1
  348. data/lib/hakumi_components/stylesheet_ownership_checker.rb +191 -0
  349. data/lib/hakumi_components/types/form_field.rb +24 -0
  350. data/lib/hakumi_components/types/html.rb +28 -0
  351. data/lib/hakumi_components/types/rendering.rb +10 -0
  352. data/lib/hakumi_components/types/stimulus.rb +19 -0
  353. data/lib/hakumi_components/types/validation.rb +22 -0
  354. data/lib/hakumi_components/types.rb +13 -0
  355. data/lib/hakumi_components/version.rb +3 -2
  356. data/lib/hakumi_components.rb +26 -5
  357. data/lib/tasks/coverage.rake +1 -0
  358. metadata +310 -346
  359. data/app/components/hakumi/admin_panel/component.rb +0 -40
  360. data/app/components/hakumi/affix/component.rb +0 -88
  361. data/app/components/hakumi/alert/component.rb +0 -84
  362. data/app/components/hakumi/anchor/component.rb +0 -80
  363. data/app/components/hakumi/anchor/link/component.rb +0 -34
  364. data/app/components/hakumi/autocomplete/component.html.erb +0 -74
  365. data/app/components/hakumi/autocomplete/component.rb +0 -202
  366. data/app/components/hakumi/avatar/component.rb +0 -91
  367. data/app/components/hakumi/badge/component.rb +0 -85
  368. data/app/components/hakumi/base_component.rb +0 -192
  369. data/app/components/hakumi/breadcrumb/component.rb +0 -27
  370. data/app/components/hakumi/breadcrumb/item/component.html.erb +0 -17
  371. data/app/components/hakumi/breadcrumb/item/component.rb +0 -35
  372. data/app/components/hakumi/button/component.rb +0 -121
  373. data/app/components/hakumi/calendar/component.rb +0 -448
  374. data/app/components/hakumi/card/component.rb +0 -94
  375. data/app/components/hakumi/card/grid/component.html.erb +0 -3
  376. data/app/components/hakumi/card/grid/component.rb +0 -37
  377. data/app/components/hakumi/card/meta/component.html.erb +0 -23
  378. data/app/components/hakumi/card/meta/component.rb +0 -46
  379. data/app/components/hakumi/carousel/component.rb +0 -147
  380. data/app/components/hakumi/cascader/component.html.erb +0 -41
  381. data/app/components/hakumi/cascader/component.rb +0 -119
  382. data/app/components/hakumi/checkbox/component.rb +0 -64
  383. data/app/components/hakumi/checkbox/group/component.html.erb +0 -12
  384. data/app/components/hakumi/checkbox/group/component.rb +0 -64
  385. data/app/components/hakumi/collapse/component.rb +0 -117
  386. data/app/components/hakumi/collapse/panel/component.rb +0 -65
  387. data/app/components/hakumi/color_picker/component.rb +0 -169
  388. data/app/components/hakumi/concerns/form_field.rb +0 -135
  389. data/app/components/hakumi/date_picker/component.rb +0 -140
  390. data/app/components/hakumi/date_picker/range_picker.rb +0 -136
  391. data/app/components/hakumi/date_picker/shared_rendering.rb +0 -199
  392. data/app/components/hakumi/descriptions/component.rb +0 -182
  393. data/app/components/hakumi/descriptions/item/component.rb +0 -28
  394. data/app/components/hakumi/divider/component.rb +0 -66
  395. data/app/components/hakumi/drawer/component.rb +0 -108
  396. data/app/components/hakumi/dropdown/component.rb +0 -97
  397. data/app/components/hakumi/dropdown/item/component.rb +0 -47
  398. data/app/components/hakumi/flex/component.rb +0 -78
  399. data/app/components/hakumi/float_button/back_top/component.rb +0 -79
  400. data/app/components/hakumi/float_button/component.rb +0 -203
  401. data/app/components/hakumi/float_button/group/component.rb +0 -227
  402. data/app/components/hakumi/float_button/group_cluster/component.rb +0 -141
  403. data/app/components/hakumi/grid/col/component.rb +0 -98
  404. data/app/components/hakumi/grid/row/component.rb +0 -79
  405. data/app/components/hakumi/image/component.rb +0 -112
  406. data/app/components/hakumi/image/preview_group/component.rb +0 -70
  407. data/app/components/hakumi/input/component.rb +0 -185
  408. data/app/components/hakumi/input/password/component.rb +0 -95
  409. data/app/components/hakumi/input/text_area/component.rb +0 -131
  410. data/app/components/hakumi/input_number/component.rb +0 -198
  411. data/app/components/hakumi/layout/component.html.erb +0 -1
  412. data/app/components/hakumi/layout/component.rb +0 -21
  413. data/app/components/hakumi/layout/content/component.html.erb +0 -1
  414. data/app/components/hakumi/layout/content/component.rb +0 -19
  415. data/app/components/hakumi/layout/footer/component.html.erb +0 -1
  416. data/app/components/hakumi/layout/footer/component.rb +0 -19
  417. data/app/components/hakumi/layout/header/component.html.erb +0 -1
  418. data/app/components/hakumi/layout/header/component.rb +0 -19
  419. data/app/components/hakumi/layout/sider/component.html.erb +0 -1
  420. data/app/components/hakumi/layout/sider/component.rb +0 -56
  421. data/app/components/hakumi/mentions/component.rb +0 -165
  422. data/app/components/hakumi/menu/component.rb +0 -126
  423. data/app/components/hakumi/menu/group/component.rb +0 -47
  424. data/app/components/hakumi/menu/item/component.rb +0 -47
  425. data/app/components/hakumi/menu/sub_menu/component.rb +0 -92
  426. data/app/components/hakumi/message/component.rb +0 -106
  427. data/app/components/hakumi/modal/component.rb +0 -99
  428. data/app/components/hakumi/modal/confirm/component.html.erb +0 -23
  429. data/app/components/hakumi/modal/confirm/component.rb +0 -46
  430. data/app/components/hakumi/modal/error/component.rb +0 -51
  431. data/app/components/hakumi/modal/info/component.rb +0 -51
  432. data/app/components/hakumi/modal/success/component.rb +0 -51
  433. data/app/components/hakumi/modal/warning/component.rb +0 -56
  434. data/app/components/hakumi/notification/component.rb +0 -126
  435. data/app/components/hakumi/pagination/component.rb +0 -205
  436. data/app/components/hakumi/popconfirm/component.rb +0 -72
  437. data/app/components/hakumi/popover/component.rb +0 -132
  438. data/app/components/hakumi/progress/component.rb +0 -565
  439. data/app/components/hakumi/qr_code/component.rb +0 -339
  440. data/app/components/hakumi/radio/component.rb +0 -80
  441. data/app/components/hakumi/radio/group/component.rb +0 -132
  442. data/app/components/hakumi/result/component.rb +0 -125
  443. data/app/components/hakumi/segmented/component.rb +0 -269
  444. data/app/components/hakumi/select/component.rb +0 -125
  445. data/app/components/hakumi/skeleton/avatar/component.rb +0 -66
  446. data/app/components/hakumi/skeleton/button/component.rb +0 -59
  447. data/app/components/hakumi/skeleton/component.rb +0 -200
  448. data/app/components/hakumi/skeleton/image/component.rb +0 -80
  449. data/app/components/hakumi/skeleton/input/component.rb +0 -51
  450. data/app/components/hakumi/skeleton/node/component.rb +0 -43
  451. data/app/components/hakumi/slider/component.rb +0 -228
  452. data/app/components/hakumi/space/compact/component.html.erb +0 -3
  453. data/app/components/hakumi/space/compact/component.rb +0 -36
  454. data/app/components/hakumi/space/component.rb +0 -73
  455. data/app/components/hakumi/spin/component.rb +0 -209
  456. data/app/components/hakumi/splitter/component.html.erb +0 -3
  457. data/app/components/hakumi/splitter/component.rb +0 -38
  458. data/app/components/hakumi/splitter/panel/component.html.erb +0 -3
  459. data/app/components/hakumi/splitter/panel/component.rb +0 -73
  460. data/app/components/hakumi/statistic/component.rb +0 -287
  461. data/app/components/hakumi/steps/component.rb +0 -145
  462. data/app/components/hakumi/steps/item/component.rb +0 -91
  463. data/app/components/hakumi/switch/component.rb +0 -124
  464. data/app/components/hakumi/table/column/component.rb +0 -80
  465. data/app/components/hakumi/table/component.rb +0 -731
  466. data/app/components/hakumi/table/concerns/columns.rb +0 -219
  467. data/app/components/hakumi/table/concerns/editable.rb +0 -121
  468. data/app/components/hakumi/table/concerns/ellipsis.rb +0 -63
  469. data/app/components/hakumi/table/concerns/fixed_columns.rb +0 -87
  470. data/app/components/hakumi/tabs/component.rb +0 -196
  471. data/app/components/hakumi/tabs/item/component.rb +0 -87
  472. data/app/components/hakumi/tag/component.rb +0 -187
  473. data/app/components/hakumi/timeline/component.html.erb +0 -7
  474. data/app/components/hakumi/timeline/component.rb +0 -168
  475. data/app/components/hakumi/timeline/item/component.rb +0 -137
  476. data/app/components/hakumi/tooltip/component.rb +0 -121
  477. data/app/components/hakumi/tour/component.rb +0 -158
  478. data/app/components/hakumi/transfer/component.html.erb +0 -173
  479. data/app/components/hakumi/transfer/component.rb +0 -245
  480. data/app/components/hakumi/tree/component.rb +0 -523
  481. data/app/components/hakumi/tree_select/component.rb +0 -261
  482. data/app/components/hakumi/typography/base_component.rb +0 -109
  483. data/app/components/hakumi/typography/link/component.rb +0 -34
  484. data/app/components/hakumi/typography/text/component.rb +0 -8
  485. data/app/components/hakumi/typography/title/component.rb +0 -29
  486. data/app/components/hakumi/upload/component.html.erb +0 -117
  487. data/app/components/hakumi/upload/component.rb +0 -193
  488. data/app/form_builders/hakumi/form_builder.rb +0 -482
  489. data/app/helpers/hakumi/form_helper.rb +0 -39
  490. data/app/services/hakumi/component_handler.rb +0 -53
  491. data/app/services/hakumi/icon/loader.rb +0 -95
  492. data/app/services/hakumi/illustrations/loader.rb +0 -43
  493. data/sig/action_view/tag_builder.rbs +0 -24
  494. data/sig/active_support/concern_patch.rbs +0 -7
  495. data/sig/generators/hakumi/install_generator.rbs +0 -11
  496. data/sig/hakumi/admin_panel/component.rbs +0 -28
  497. data/sig/hakumi/affix/component.rbs +0 -44
  498. data/sig/hakumi/alert/component.rbs +0 -64
  499. data/sig/hakumi/anchor/component.rbs +0 -45
  500. data/sig/hakumi/anchor/link/component.rbs +0 -31
  501. data/sig/hakumi/autocomplete/component.rbs +0 -68
  502. data/sig/hakumi/avatar/component.rbs +0 -44
  503. data/sig/hakumi/badge/component.rbs +0 -41
  504. data/sig/hakumi/base_component.rbs +0 -82
  505. data/sig/hakumi/breadcrumb/component.rbs +0 -21
  506. data/sig/hakumi/breadcrumb/item/component.rbs +0 -36
  507. data/sig/hakumi/button/component.rbs +0 -56
  508. data/sig/hakumi/calendar/component.rbs +0 -41
  509. data/sig/hakumi/card/component.rbs +0 -50
  510. data/sig/hakumi/card/grid/component.rbs +0 -21
  511. data/sig/hakumi/card/meta/component.rbs +0 -25
  512. data/sig/hakumi/carousel/component.rbs +0 -88
  513. data/sig/hakumi/cascader/component.rbs +0 -52
  514. data/sig/hakumi/checkbox/component.rbs +0 -46
  515. data/sig/hakumi/checkbox/group/component.rbs +0 -34
  516. data/sig/hakumi/collapse/component.rbs +0 -57
  517. data/sig/hakumi/collapse/panel/component.rbs +0 -47
  518. data/sig/hakumi/color_picker/component.rbs +0 -80
  519. data/sig/hakumi/concerns/form_field.rbs +0 -89
  520. data/sig/hakumi/container/component.rbs +0 -37
  521. data/sig/hakumi/date_picker/component.rbs +0 -72
  522. data/sig/hakumi/date_picker/range_picker.rbs +0 -68
  523. data/sig/hakumi/date_picker/shared_rendering.rbs +0 -41
  524. data/sig/hakumi/descriptions/component.rbs +0 -59
  525. data/sig/hakumi/descriptions/item/component.rbs +0 -31
  526. data/sig/hakumi/divider/component.rbs +0 -39
  527. data/sig/hakumi/drawer/component.rbs +0 -66
  528. data/sig/hakumi/dropdown/component.rbs +0 -45
  529. data/sig/hakumi/dropdown/divider/component.rbs +0 -11
  530. data/sig/hakumi/dropdown/item/component.rbs +0 -45
  531. data/sig/hakumi/empty/component.rbs +0 -40
  532. data/sig/hakumi/flex/component.rbs +0 -47
  533. data/sig/hakumi/float_button/back_top/component.rbs +0 -45
  534. data/sig/hakumi/float_button/component.rbs +0 -80
  535. data/sig/hakumi/float_button/group/component.rbs +0 -95
  536. data/sig/hakumi/float_button/group_cluster/component.rbs +0 -60
  537. data/sig/hakumi/form/item/component.rbs +0 -39
  538. data/sig/hakumi/form_builder.rbs +0 -37
  539. data/sig/hakumi/grid/col/component.rbs +0 -50
  540. data/sig/hakumi/grid/row/component.rbs +0 -46
  541. data/sig/hakumi/icon/component.rbs +0 -56
  542. data/sig/hakumi/image/component.rbs +0 -56
  543. data/sig/hakumi/image/preview_group/component.rbs +0 -28
  544. data/sig/hakumi/input/component.rbs +0 -71
  545. data/sig/hakumi/input/password/component.rbs +0 -27
  546. data/sig/hakumi/input/text_area/component.rbs +0 -42
  547. data/sig/hakumi/input_number/component.rbs +0 -85
  548. data/sig/hakumi/layout/component.rbs +0 -21
  549. data/sig/hakumi/layout/content/component.rbs +0 -19
  550. data/sig/hakumi/layout/footer/component.rbs +0 -19
  551. data/sig/hakumi/layout/header/component.rbs +0 -19
  552. data/sig/hakumi/layout/sider/component.rbs +0 -19
  553. data/sig/hakumi/mentions/component.rbs +0 -75
  554. data/sig/hakumi/menu/component.rbs +0 -52
  555. data/sig/hakumi/menu/divider/component.rbs +0 -11
  556. data/sig/hakumi/menu/group/component.rbs +0 -20
  557. data/sig/hakumi/menu/item/component.rbs +0 -42
  558. data/sig/hakumi/menu/sub_menu/component.rbs +0 -37
  559. data/sig/hakumi/message/component.rbs +0 -50
  560. data/sig/hakumi/modal/component.rbs +0 -76
  561. data/sig/hakumi/modal/confirm/component.rbs +0 -32
  562. data/sig/hakumi/modal/error/component.rbs +0 -27
  563. data/sig/hakumi/modal/info/component.rbs +0 -27
  564. data/sig/hakumi/modal/success/component.rbs +0 -27
  565. data/sig/hakumi/modal/warning/component.rbs +0 -27
  566. data/sig/hakumi/notification/component.rbs +0 -64
  567. data/sig/hakumi/pagination/component.rbs +0 -113
  568. data/sig/hakumi/popconfirm/component.rbs +0 -52
  569. data/sig/hakumi/popover/component.rbs +0 -66
  570. data/sig/hakumi/progress/component.rbs +0 -201
  571. data/sig/hakumi/qr_code/component.rbs +0 -111
  572. data/sig/hakumi/radio/component.rbs +0 -55
  573. data/sig/hakumi/radio/group/component.rbs +0 -71
  574. data/sig/hakumi/rails/attribute_introspection.rbs +0 -16
  575. data/sig/hakumi/rails/validation_introspection.rbs +0 -18
  576. data/sig/hakumi/rails/validation_mapper.rbs +0 -53
  577. data/sig/hakumi/rails.rbs +0 -6
  578. data/sig/hakumi/rate/component.rbs +0 -56
  579. data/sig/hakumi/result/component.rbs +0 -65
  580. data/sig/hakumi/segmented/component.rbs +0 -72
  581. data/sig/hakumi/select/component.rbs +0 -73
  582. data/sig/hakumi/skeleton/avatar/component.rbs +0 -23
  583. data/sig/hakumi/skeleton/button/component.rbs +0 -24
  584. data/sig/hakumi/skeleton/component.rbs +0 -76
  585. data/sig/hakumi/skeleton/image/component.rbs +0 -22
  586. data/sig/hakumi/skeleton/input/component.rbs +0 -23
  587. data/sig/hakumi/skeleton/node/component.rbs +0 -21
  588. data/sig/hakumi/slider/component.rbs +0 -85
  589. data/sig/hakumi/space/compact/component.rbs +0 -21
  590. data/sig/hakumi/space/component.rbs +0 -47
  591. data/sig/hakumi/spin/component.rbs +0 -77
  592. data/sig/hakumi/splitter/component.rbs +0 -27
  593. data/sig/hakumi/splitter/panel/component.rbs +0 -36
  594. data/sig/hakumi/statistic/component.rbs +0 -97
  595. data/sig/hakumi/steps/component.rbs +0 -82
  596. data/sig/hakumi/steps/item/component.rbs +0 -29
  597. data/sig/hakumi/switch/component.rbs +0 -52
  598. data/sig/hakumi/table/column/component.rbs +0 -57
  599. data/sig/hakumi/table/column_group/component.rbs +0 -20
  600. data/sig/hakumi/table/component.rbs +0 -279
  601. data/sig/hakumi/table/concerns/columns.rbs +0 -95
  602. data/sig/hakumi/table/concerns/editable.rbs +0 -40
  603. data/sig/hakumi/table/concerns/ellipsis.rbs +0 -27
  604. data/sig/hakumi/table/concerns/fixed_columns.rbs +0 -33
  605. data/sig/hakumi/tabs/component.rbs +0 -66
  606. data/sig/hakumi/tabs/item/component.rbs +0 -29
  607. data/sig/hakumi/tag/component.rbs +0 -86
  608. data/sig/hakumi/tag/group/component.rbs +0 -24
  609. data/sig/hakumi/time_picker/component.rbs +0 -72
  610. data/sig/hakumi/timeline/component.rbs +0 -61
  611. data/sig/hakumi/timeline/item/component.rbs +0 -70
  612. data/sig/hakumi/tooltip/component.rbs +0 -73
  613. data/sig/hakumi/tour/component.rbs +0 -84
  614. data/sig/hakumi/transfer/component.rbs +0 -75
  615. data/sig/hakumi/tree/component.rbs +0 -126
  616. data/sig/hakumi/tree_select/component.rbs +0 -122
  617. data/sig/hakumi/typography/base_component.rbs +0 -57
  618. data/sig/hakumi/typography/link/component.rbs +0 -28
  619. data/sig/hakumi/typography/paragraph/component.rbs +0 -13
  620. data/sig/hakumi/typography/title/component.rbs +0 -28
  621. data/sig/hakumi/upload/component.rbs +0 -78
  622. data/sig/hakumi.rbs +0 -96
  623. data/sig/rails/active_model/validations/comparison_validator.rbs +0 -6
  624. data/sig/rails.rbs +0 -18
  625. data/sig/view_component/base.rbs +0 -28
  626. /data/app/components/{hakumi → hakumi_components}/affix/component.html.erb +0 -0
  627. /data/app/components/{hakumi → hakumi_components}/anchor/component.html.erb +0 -0
  628. /data/app/components/{hakumi → hakumi_components}/breadcrumb/component.html.erb +0 -0
  629. /data/app/components/{hakumi → hakumi_components}/dropdown/component.html.erb +0 -0
  630. /data/app/components/{hakumi → hakumi_components}/dropdown/item/component.html.erb +0 -0
  631. /data/app/components/{hakumi → hakumi_components}/icon/component.html.erb +0 -0
  632. /data/app/components/{hakumi → hakumi_components}/menu/component.html.erb +0 -0
  633. /data/app/components/{hakumi → hakumi_components}/menu/group/component.html.erb +0 -0
  634. /data/app/components/{hakumi → hakumi_components}/menu/item/component.html.erb +0 -0
  635. /data/app/components/{hakumi → hakumi_components}/menu/sub_menu/component.html.erb +0 -0
  636. /data/app/components/{hakumi → hakumi_components}/message/component.html.erb +0 -0
  637. /data/app/components/{hakumi → hakumi_components}/qr_code/component.html.erb +0 -0
  638. /data/app/components/{hakumi → hakumi_components}/result/component.html.erb +0 -0
  639. /data/app/components/{hakumi → hakumi_components}/skeleton/avatar/component.html.erb +0 -0
  640. /data/app/components/{hakumi → hakumi_components}/skeleton/button/component.html.erb +0 -0
  641. /data/app/components/{hakumi → hakumi_components}/skeleton/component.html.erb +0 -0
  642. /data/app/components/{hakumi → hakumi_components}/skeleton/input/component.html.erb +0 -0
  643. /data/app/components/{hakumi → hakumi_components}/skeleton/node/component.html.erb +0 -0
  644. /data/app/components/{hakumi → hakumi_components}/spin/component.html.erb +0 -0
  645. /data/app/components/{hakumi → hakumi_components}/statistic/component.html.erb +0 -0
  646. /data/app/components/{hakumi → hakumi_components}/steps/item/component.html.erb +0 -0
  647. /data/app/components/{hakumi → hakumi_components}/tour/component.html.erb +0 -0
@@ -0,0 +1,149 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Progress
6
+ class StepsRenderer
7
+ extend T::Sig
8
+
9
+ ProgressValue = T.type_alias { Numeric }
10
+ StrokeGradient = T.type_alias { T::Hash[String, String] }
11
+ StrokeColor = T.type_alias { T.nilable(T.any(String, StrokeGradient)) }
12
+ StepStateValue = T.type_alias { Types::HtmlPrimitive }
13
+ StepState = T.type_alias { T::Hash[Symbol, StepStateValue] }
14
+
15
+ CIRCLE_STEP_GAP = T.let(2, Integer)
16
+
17
+ sig do
18
+ params(
19
+ type: Symbol,
20
+ percent: ProgressValue,
21
+ steps: T.nilable(Integer),
22
+ stroke_color: StrokeColor,
23
+ circle_geometry: T.nilable(CircleGeometry),
24
+ gradient_id: T.nilable(String)
25
+ ).void
26
+ end
27
+ def initialize(type:, percent:, steps:, stroke_color:, circle_geometry: nil, gradient_id: nil)
28
+ @type = T.let(type, Symbol)
29
+ @percent = T.let(percent, ProgressValue)
30
+ @steps = T.let(steps, T.nilable(Integer))
31
+ @stroke_color = T.let(stroke_color, StrokeColor)
32
+ @circle_geometry = T.let(circle_geometry, T.nilable(CircleGeometry))
33
+ @gradient_id = T.let(gradient_id, T.nilable(String))
34
+ end
35
+
36
+ sig { returns(T::Boolean) }
37
+ def circle_steps?
38
+ (circle? || dashboard?) && steps?
39
+ end
40
+
41
+ sig { returns(T::Array[StepState]) }
42
+ def line_steps
43
+ return [] unless line? && steps?
44
+
45
+ total = step_count
46
+ progress_units = (@percent.to_f / 100.0) * total
47
+
48
+ (0...total).map do |index|
49
+ {
50
+ active: progress_units > index,
51
+ current: index == progress_units.floor && progress_units < total,
52
+ width: "#{(100.0 / total).round(2)}%",
53
+ custom_color: line_custom_color
54
+ }
55
+ end
56
+ end
57
+
58
+ sig { returns(T::Array[StepState]) }
59
+ def circle_steps
60
+ return [] unless circle_steps?
61
+
62
+ geometry = @circle_geometry
63
+ return [] if geometry.nil?
64
+
65
+ total = step_count
66
+ effective = geometry.effective_length
67
+ segment_length = (effective - (CIRCLE_STEP_GAP * total)) / total.to_f
68
+ progress_units = (@percent.to_f / 100.0) * total
69
+ base_offset = format_float(geometry.dashoffset).to_f
70
+
71
+ (0...total).map do |index|
72
+ active = progress_units > index
73
+ current = index == progress_units.floor && progress_units < total
74
+ offset = base_offset + ((segment_length + CIRCLE_STEP_GAP) * index)
75
+ {
76
+ index: index,
77
+ active: active,
78
+ current: current,
79
+ dasharray: geometry.dasharray(segment_length),
80
+ dashoffset: format_float(offset),
81
+ stroke: circle_step_stroke(active)
82
+ }
83
+ end
84
+ end
85
+
86
+ private
87
+
88
+ sig { returns(T::Boolean) }
89
+ def line?
90
+ @type == :line
91
+ end
92
+
93
+ sig { returns(T::Boolean) }
94
+ def circle?
95
+ @type == :circle
96
+ end
97
+
98
+ sig { returns(T::Boolean) }
99
+ def dashboard?
100
+ @type == :dashboard
101
+ end
102
+
103
+ sig { returns(T::Boolean) }
104
+ def steps?
105
+ !@steps.nil? && @steps.positive?
106
+ end
107
+
108
+ sig { returns(Integer) }
109
+ def step_count
110
+ steps = @steps
111
+ return 0 if steps.nil?
112
+
113
+ steps
114
+ end
115
+
116
+ sig { returns(T.nilable(String)) }
117
+ def line_custom_color
118
+ color = @stroke_color
119
+ return nil unless color.is_a?(String)
120
+ return nil if color.empty?
121
+
122
+ color
123
+ end
124
+
125
+ sig { params(active: T::Boolean).returns(T.nilable(String)) }
126
+ def circle_step_stroke(active)
127
+ return "var(--hakumi-progress-trail-color, var(--color-border-secondary))" unless active
128
+
129
+ gradient_id = @gradient_id
130
+ return "url(##{gradient_id})" if gradient? && !gradient_id.nil?
131
+
132
+ color = @stroke_color
133
+ return color if color.is_a?(String) && !color.empty?
134
+
135
+ nil
136
+ end
137
+
138
+ sig { returns(T::Boolean) }
139
+ def gradient?
140
+ @stroke_color.is_a?(Hash)
141
+ end
142
+
143
+ sig { params(value: Float).returns(String) }
144
+ def format_float(value)
145
+ format("%.2f", value)
146
+ end
147
+ end
148
+ end
149
+ end
@@ -0,0 +1,90 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Progress
6
+ class StrokeGradientValue
7
+ extend T::Sig
8
+
9
+ StrokeGradient = T.type_alias { T::Hash[String, String] }
10
+ StrokeColor = T.type_alias { T.nilable(T.any(String, StrokeGradient)) }
11
+
12
+ sig { params(source: T::Hash[T.untyped, T.untyped], compact_blank: T::Boolean).returns(StrokeColor) }
13
+ def self.normalize(source, compact_blank: true)
14
+ gradient = T.let({}, StrokeGradient)
15
+ source.each do |key, value|
16
+ next if value.nil?
17
+
18
+ string_value = compact_blank ? value.to_s.strip : value.to_s
19
+ next if compact_blank && string_value.empty?
20
+
21
+ gradient[key.to_s] = string_value
22
+ end
23
+ gradient.empty? ? nil : gradient
24
+ end
25
+
26
+ sig { returns(Types::HtmlAttributes) }
27
+ def self.default_svg_direction
28
+ { x1: "0%", y1: "0%", x2: "100%", y2: "0%" }
29
+ end
30
+
31
+ sig { params(stops: T::Array[[ String, String ]]).returns(String) }
32
+ def self.stop_list(stops)
33
+ stops.map { |key, color| "#{color} #{key}" }.join(", ")
34
+ end
35
+
36
+ sig { params(source: StrokeGradient).void }
37
+ def initialize(source)
38
+ @source = T.let(source, StrokeGradient)
39
+ end
40
+
41
+ sig { returns(String) }
42
+ def direction
43
+ @source["direction"] || "to right"
44
+ end
45
+
46
+ sig { returns(T::Array[[ String, String ]]) }
47
+ def stop_pairs
48
+ entries = @source.reject { |key, value| key == "direction" || value.nil? }
49
+
50
+ if entries.key?("from") || entries.key?("to")
51
+ pairs = T.let([], T::Array[[ String, String ]])
52
+ from = entries["from"]
53
+ to = entries["to"]
54
+ pairs << [ "0%", from ] if from
55
+ pairs << [ "100%", to ] if to
56
+ return pairs
57
+ end
58
+
59
+ entries.map { |key, value| [ key, value ] }
60
+ end
61
+
62
+ sig { returns(T.nilable(String)) }
63
+ def css
64
+ stops = stop_pairs
65
+ return nil if stops.empty?
66
+
67
+ "linear-gradient(#{direction}, #{stop_list(stops)})"
68
+ end
69
+
70
+ sig { returns(Types::HtmlAttributes) }
71
+ def svg_direction
72
+ case direction
73
+ when "to left"
74
+ { x1: "100%", y1: "0%", x2: "0%", y2: "0%" }
75
+ when "to bottom"
76
+ { x1: "0%", y1: "0%", x2: "0%", y2: "100%" }
77
+ when "to top"
78
+ { x1: "0%", y1: "100%", x2: "0%", y2: "0%" }
79
+ else
80
+ self.class.default_svg_direction
81
+ end
82
+ end
83
+
84
+ sig { params(stops: T::Array[[ String, String ]]).returns(String) }
85
+ def stop_list(stops)
86
+ self.class.stop_list(stops)
87
+ end
88
+ end
89
+ end
90
+ end
@@ -0,0 +1,405 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module QrCode
6
+ class Component < HakumiComponents::BaseComponent
7
+ extend T::Sig
8
+
9
+ NumberValue = T.type_alias { Numeric }
10
+ NumberInput = T.type_alias { BaseComponent::DimensionInput }
11
+ OptionInput = T.type_alias { BaseComponent::SymbolInput }
12
+ IconRenderable = T.type_alias { T.nilable(Types::ComponentRenderable) }
13
+ IconInput = T.type_alias { T.nilable(T.any(Symbol, Types::Renderable)) }
14
+ StatusRender = T.type_alias { T.nilable(T.any(Types::Renderable, T.proc.params(status: Symbol).returns(Types::RenderedContent))) }
15
+ ControllerOptions = T.type_alias { BaseComponent::ControllerOptions }
16
+ ImageAttributes = T.type_alias { T::Hash[Symbol, Types::HtmlPrimitive] }
17
+ TYPES = T.let([ :canvas, :svg ].freeze, T::Array[Symbol])
18
+ STATUSES = T.let([ :active, :expired, :loading, :scanned ].freeze, T::Array[Symbol])
19
+ ERROR_LEVELS = T.let([ :L, :M, :Q, :H ].freeze, T::Array[Symbol])
20
+
21
+ sig do
22
+ params(params: ControllerOptions).returns(Types::HtmlAttributes)
23
+ end
24
+ def self.extract_controller_locals(params)
25
+ bordered_value = html_param(params, :bordered)
26
+
27
+ {
28
+ value: string_html_param(html_param(params, :value)),
29
+ size: parse_number(html_param(params, :size)),
30
+ type: symbol_html_param(html_param(params, :type)),
31
+ color: string_html_param(html_param(params, :color)),
32
+ bg_color: string_html_param(html_param(params, :bg_color)),
33
+ icon: html_param(params, :icon),
34
+ icon_size: parse_number(html_param(params, :icon_size)),
35
+ status: symbol_html_param(html_param(params, :status)),
36
+ status_render: string_html_param(html_param(params, :status_render)),
37
+ bordered: bordered_value.nil? ? nil : boolean_html_param(bordered_value),
38
+ error_level: parse_error_level(html_param(params, :error_level)),
39
+ download_name: string_html_param(html_param(params, :download_name)),
40
+ aria_label: string_html_param(html_param(params, :aria_label))
41
+ }
42
+ end
43
+
44
+ sig { params(value: Types::HtmlAttributeValue).returns(T.nilable(NumberValue)) }
45
+ def self.parse_number(value)
46
+ return nil if value.nil?
47
+ return value if value.is_a?(Integer)
48
+ return value if value.is_a?(Float)
49
+ raw = value.to_s.strip
50
+ return nil if raw.empty?
51
+
52
+ raw.include?(".") ? raw.to_f : raw.to_i
53
+ end
54
+
55
+ sig { params(value: Types::HtmlAttributeValue).returns(T.nilable(Symbol)) }
56
+ def self.parse_error_level(value)
57
+ return nil if value.nil?
58
+
59
+ value.to_s.strip.presence&.upcase&.to_sym
60
+ end
61
+
62
+ sig do
63
+ params(
64
+ value: String,
65
+ size: NumberInput,
66
+ type: OptionInput,
67
+ color: T.nilable(String),
68
+ bg_color: T.nilable(String),
69
+ icon: IconInput,
70
+ icon_size: NumberInput,
71
+ status: OptionInput,
72
+ status_render: StatusRender,
73
+ bordered: T.nilable(T::Boolean),
74
+ error_level: OptionInput,
75
+ download_name: T.nilable(String),
76
+ aria_label: T.nilable(String),
77
+ html_options: Types::HtmlAttributeValue
78
+ ).void
79
+ end
80
+ def initialize(
81
+ value:,
82
+ size: 160,
83
+ type: :canvas,
84
+ color: nil,
85
+ bg_color: nil,
86
+ icon: nil,
87
+ icon_size: 40,
88
+ status: :active,
89
+ status_render: nil,
90
+ bordered: true,
91
+ error_level: :M,
92
+ download_name: nil,
93
+ aria_label: nil,
94
+ **html_options
95
+ )
96
+ @value = T.let(value, String)
97
+ @size = T.let(self.class.parse_number(size) || 160, NumberValue)
98
+ @type = T.let(normalize_symbol(type) || :canvas, Symbol)
99
+ @color = T.let(normalize_string(color), T.nilable(String))
100
+ @bg_color = T.let(normalize_string(bg_color), T.nilable(String))
101
+ @icon_input = T.let(icon, IconInput)
102
+ @icon_size = T.let(self.class.parse_number(icon_size) || 40, NumberValue)
103
+ resolve_icon!
104
+ @status = T.let(normalize_symbol(status) || :active, Symbol)
105
+ @status_render = T.let(status_render, StatusRender)
106
+ @bordered = T.let(bordered.nil? ? true : cast_boolean(bordered) == true, T::Boolean)
107
+ @error_level = T.let(normalize_error_level(error_level) || :M, Symbol)
108
+ @download_name = T.let(normalize_string(download_name), T.nilable(String))
109
+ @aria_label = T.let(normalize_string(aria_label), T.nilable(String))
110
+ @html_options = T.let(html_options, Types::HtmlAttributes)
111
+
112
+ validate_props!
113
+ end
114
+
115
+ sig { returns(Types::HtmlAttributes) }
116
+ def wrapper_attributes
117
+ merge_attributes(
118
+ {
119
+ class: wrapper_classes,
120
+ style: wrapper_styles,
121
+ role: "img",
122
+ "aria-label": @aria_label || t_default(:aria_label, default: "QR Code"),
123
+ "aria-busy": loading? ? "true" : "false",
124
+ data: data_attributes
125
+ },
126
+ @html_options
127
+ )
128
+ end
129
+
130
+ sig { returns(Types::HtmlAttributes) }
131
+ def renderer_attributes
132
+ T.let({
133
+ class: "hakumi-qr-code-inner",
134
+ data: { "hakumi--qr-code-target": "renderer" }
135
+ }, Types::HtmlAttributes)
136
+ end
137
+
138
+ sig { returns(T::Boolean) }
139
+ def icon?
140
+ icon_component? || icon_image?
141
+ end
142
+
143
+ sig { returns(Types::RenderedContent) }
144
+ def icon_markup
145
+ icon_component = @icon_component
146
+ if icon_image?
147
+ tag.img(**icon_image_attributes)
148
+ elsif !icon_component.nil?
149
+ render(icon_component)
150
+ end
151
+ end
152
+
153
+ sig { returns(ImageAttributes) }
154
+ def icon_image_attributes
155
+ T.let({
156
+ src: @icon_url,
157
+ alt: "",
158
+ class: "hakumi-qr-code-icon-image",
159
+ "aria-hidden": true,
160
+ role: "presentation"
161
+ }, ImageAttributes)
162
+ end
163
+
164
+ sig { returns(T::Boolean) }
165
+ def status_overlay?
166
+ @status != :active
167
+ end
168
+
169
+ sig { returns(Types::RenderedContent) }
170
+ def status_overlay_markup
171
+ return nil unless status_overlay?
172
+
173
+ custom = custom_status_overlay
174
+ return custom if custom
175
+
176
+ default_status_overlay
177
+ end
178
+
179
+ sig { returns(String) }
180
+ def status_text
181
+ key = "status.#{@status}"
182
+ default = case @status
183
+ when :expired then "Expired"
184
+ when :loading then "Loading"
185
+ when :scanned then "Scanned"
186
+ else "Active"
187
+ end
188
+ t_default(key, default: default)
189
+ end
190
+
191
+ sig { returns(T.nilable(String)) }
192
+ def status_description
193
+ case @status
194
+ when :expired
195
+ t_default("description.expired", default: "Please refresh to continue")
196
+ when :scanned
197
+ t_default("description.scanned", default: "You can now continue")
198
+ else
199
+ nil
200
+ end
201
+ end
202
+
203
+ private
204
+
205
+ sig { void }
206
+ def validate_props!
207
+ validate_required!(:value)
208
+ validate_inclusion!(:type, TYPES)
209
+ validate_inclusion!(:status, STATUSES)
210
+ validate_inclusion!(:error_level, ERROR_LEVELS)
211
+
212
+ if @size.to_f <= 0
213
+ raise ArgumentError, "size must be a positive number"
214
+ end
215
+
216
+ if icon? && @icon_size.to_f <= 0
217
+ raise ArgumentError, "icon_size must be a positive number"
218
+ end
219
+ end
220
+
221
+ sig { returns(String) }
222
+ def wrapper_classes
223
+ modifiers = {
224
+ @type => true,
225
+ "status-#{@status}" => true,
226
+ bordered: @bordered,
227
+ "with-icon" => icon?,
228
+ "custom-status" => custom_status_render?
229
+ }
230
+
231
+ component_classes("qr-code", modifiers, @html_options)
232
+ end
233
+
234
+ sig { returns(String) }
235
+ def wrapper_styles
236
+ styles = {
237
+ "--hakumi-qr-code-size" => size_css_value,
238
+ "--hakumi-qr-code-icon-size" => icon_css_value,
239
+ "--hakumi-qr-code-color" => @color,
240
+ "--hakumi-qr-code-bg-color" => @bg_color
241
+ }
242
+
243
+ styles.compact.map { |key, value| "#{key}: #{value}" }.join("; ")
244
+ end
245
+
246
+ sig { returns(String) }
247
+ def size_css_value
248
+ "#{formatted_number(@size)}px"
249
+ end
250
+
251
+ sig { returns(T.nilable(String)) }
252
+ def icon_css_value
253
+ return nil unless icon?
254
+
255
+ "#{formatted_number(@icon_size)}px"
256
+ end
257
+
258
+ sig { void }
259
+ def resolve_icon!
260
+ @icon_component = T.let(nil, IconRenderable)
261
+ @icon_url = T.let(nil, T.nilable(String))
262
+
263
+ if @icon_input.is_a?(ViewComponent::Base) || @icon_input.is_a?(ViewComponent::Slot)
264
+ @icon_component = @icon_input
265
+ return
266
+ end
267
+
268
+ if @icon_input.is_a?(String)
269
+ trimmed = @icon_input.strip
270
+ if trimmed.downcase.start_with?("http://", "https://", "data:")
271
+ @icon_url = trimmed
272
+ return
273
+ end
274
+ end
275
+
276
+ icon_name = extract_icon_name(@icon_input)
277
+ return unless icon_name
278
+
279
+ @icon_component = HakumiComponents::Icon::Component.new(name: icon_name, size: icon_component_size)
280
+ end
281
+
282
+ sig { returns(T::Boolean) }
283
+ def icon_component?
284
+ !@icon_component.nil?
285
+ end
286
+
287
+ sig { returns(T::Boolean) }
288
+ def icon_image?
289
+ @icon_url.present?
290
+ end
291
+
292
+ sig { returns(Types::DataAttributes) }
293
+ def data_attributes
294
+ data = T.let({
295
+ controller: "hakumi--qr-code",
296
+ "hakumi--qr-code-value-value": @value,
297
+ "hakumi--qr-code-size-value": @size,
298
+ "hakumi--qr-code-type-value": @type,
299
+ "hakumi--qr-code-color-value": @color,
300
+ "hakumi--qr-code-bg-color-value": @bg_color,
301
+ "hakumi--qr-code-error-level-value": @error_level,
302
+ "hakumi--qr-code-status-value": @status
303
+ }, Types::DataAttributes)
304
+
305
+ data["hakumi--qr-code-download-name-value"] = @download_name if @download_name.present?
306
+ data
307
+ end
308
+
309
+ sig { returns(T::Boolean) }
310
+ def loading?
311
+ @status == :loading
312
+ end
313
+
314
+ sig { returns(T::Boolean) }
315
+ def custom_status_render?
316
+ @status_render.present?
317
+ end
318
+
319
+ sig { returns(Types::RenderedContent) }
320
+ def custom_status_overlay
321
+ return unless custom_status_render?
322
+
323
+ if @status_render.is_a?(Proc)
324
+ return view_context.capture { @status_render.call(@status) }
325
+ end
326
+
327
+ if @status_render.is_a?(ViewComponent::Base) || @status_render.is_a?(ViewComponent::Slot)
328
+ return render(@status_render)
329
+ end
330
+
331
+ @status_render
332
+ end
333
+
334
+ sig { returns(ActiveSupport::SafeBuffer) }
335
+ def default_status_overlay
336
+ content = T.let([], T::Array[Types::RenderedContent])
337
+ if loading?
338
+ content << render(HakumiComponents::Spin::Component.new(size: :small))
339
+ end
340
+
341
+ content << tag.span(status_text, class: "hakumi-qr-code-status-text")
342
+ if status_description.present?
343
+ content << tag.span(status_description, class: "hakumi-qr-code-status-description")
344
+ end
345
+
346
+ tag.div(class: "hakumi-qr-code-status-content") do
347
+ safe_join(content)
348
+ end
349
+ end
350
+
351
+ sig { params(value: OptionInput).returns(T.nilable(Symbol)) }
352
+ def normalize_symbol(value)
353
+ return nil if value.nil?
354
+
355
+ raw = value.is_a?(String) ? value.dup : value.to_s.dup
356
+ raw.strip!
357
+ return nil if raw.empty?
358
+
359
+ raw.to_sym
360
+ end
361
+
362
+ sig { params(value: OptionInput).returns(T.nilable(Symbol)) }
363
+ def normalize_error_level(value)
364
+ symbol = normalize_symbol(value)
365
+ symbol&.to_s&.upcase&.to_sym
366
+ end
367
+
368
+ sig { params(value: T.nilable(Types::ValidationComparable)).returns(T.nilable(String)) }
369
+ def normalize_string(value)
370
+ return nil if value.nil?
371
+
372
+ raw = value.to_s.strip
373
+ raw.empty? ? nil : raw
374
+ end
375
+
376
+ sig { params(value: IconInput).returns(T.nilable(Symbol)) }
377
+ def extract_icon_name(value)
378
+ case value
379
+ when Symbol
380
+ value
381
+ when String
382
+ trimmed = value.strip
383
+ return nil if trimmed.empty?
384
+ trimmed.to_sym
385
+ else
386
+ nil
387
+ end
388
+ end
389
+
390
+ sig { returns(Integer) }
391
+ def icon_component_size
392
+ size = (@icon_size.to_f * 0.65).round
393
+ size.positive? ? size : 24
394
+ end
395
+
396
+ sig { params(value: NumberValue).returns(NumberValue) }
397
+ def formatted_number(value)
398
+ return value if value.is_a?(Integer)
399
+
400
+ remainder = value % 1
401
+ remainder.zero? ? value.to_i : value
402
+ end
403
+ end
404
+ end
405
+ end
@@ -2,15 +2,15 @@
2
2
  <label class="<%= wrapper_classes %>" data-controller="<%= data_controllers %>" data-hakumi--radio-checked-value="<%= @checked %>" data-hakumi--radio-variant-value="default">
3
3
  <span class="<%= radio_classes %>" data-hakumi--radio-target="radio">
4
4
  <%= radio_button_tag(
5
- @name,
5
+ form_field_name,
6
6
  @value,
7
7
  @checked,
8
8
  id: input_id,
9
9
  disabled: @disabled,
10
10
  autofocus: @auto_focus,
11
- required: @required,
11
+ required: form_field_required,
12
12
  class: "hakumi-radio-input",
13
- "aria-invalid": has_error? ? "true" : nil,
13
+ "aria-invalid": error? ? "true" : nil,
14
14
  "aria-describedby": describedby_ids,
15
15
  data: (@html_options[:data] || {}).merge({
16
16
  hakumi__radio_target: "input",