hakumi_components 0.1.18.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 (467) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +32 -2
  3. data/README.md +208 -381
  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_components/admin_panel/component.rb +19 -9
  7. data/app/components/hakumi_components/affix/component.rb +54 -30
  8. data/app/components/hakumi_components/alert/component.html.erb +2 -2
  9. data/app/components/hakumi_components/alert/component.rb +68 -18
  10. data/app/components/hakumi_components/anchor/component.rb +35 -27
  11. data/app/components/hakumi_components/anchor/link/component.html.erb +1 -1
  12. data/app/components/hakumi_components/anchor/link/component.rb +28 -9
  13. data/app/components/hakumi_components/autocomplete/component.html.erb +86 -66
  14. data/app/components/hakumi_components/autocomplete/component.rb +115 -89
  15. data/app/components/hakumi_components/avatar/component.rb +45 -12
  16. data/app/components/hakumi_components/badge/component.rb +99 -45
  17. data/app/components/hakumi_components/base_component.rb +290 -91
  18. data/app/components/hakumi_components/breadcrumb/component.rb +16 -5
  19. data/app/components/hakumi_components/breadcrumb/item/component.html.erb +5 -5
  20. data/app/components/hakumi_components/breadcrumb/item/component.rb +56 -8
  21. data/app/components/hakumi_components/breadcrumb/item/menu_entry.rb +71 -0
  22. data/app/components/hakumi_components/button/component.rb +62 -20
  23. data/app/components/hakumi_components/calendar/component.html.erb +19 -19
  24. data/app/components/hakumi_components/calendar/component.rb +225 -202
  25. data/app/components/hakumi_components/calendar/day_cell.rb +62 -0
  26. data/app/components/hakumi_components/calendar/display_model.rb +237 -0
  27. data/app/components/hakumi_components/calendar/event_entry.rb +54 -0
  28. data/app/components/hakumi_components/calendar/locale_pack.rb +28 -0
  29. data/app/components/hakumi_components/calendar/panel_entry.rb +48 -0
  30. data/app/components/hakumi_components/calendar/week_row.rb +28 -0
  31. data/app/components/hakumi_components/card/component.html.erb +2 -2
  32. data/app/components/hakumi_components/card/component.rb +76 -49
  33. data/app/components/hakumi_components/card/grid/component.html.erb +1 -1
  34. data/app/components/hakumi_components/card/grid/component.rb +23 -17
  35. data/app/components/hakumi_components/card/meta/component.html.erb +8 -8
  36. data/app/components/hakumi_components/card/meta/component.rb +42 -24
  37. data/app/components/hakumi_components/carousel/autoplay_config.rb +27 -0
  38. data/app/components/hakumi_components/carousel/component.rb +85 -67
  39. data/app/components/hakumi_components/carousel/dots_config.rb +36 -0
  40. data/app/components/hakumi_components/cascader/component.html.erb +54 -36
  41. data/app/components/hakumi_components/cascader/component.rb +111 -62
  42. data/app/components/hakumi_components/checkbox/component.html.erb +5 -5
  43. data/app/components/hakumi_components/checkbox/component.rb +40 -6
  44. data/app/components/hakumi_components/checkbox/group/component.html.erb +4 -4
  45. data/app/components/hakumi_components/checkbox/group/component.rb +40 -13
  46. data/app/components/hakumi_components/checkbox/group/option.rb +34 -0
  47. data/app/components/hakumi_components/collapse/component.rb +64 -48
  48. data/app/components/hakumi_components/collapse/panel/component.rb +64 -21
  49. data/app/components/hakumi_components/color_picker/component.html.erb +21 -22
  50. data/app/components/hakumi_components/color_picker/component.rb +159 -49
  51. data/app/components/hakumi_components/color_picker/preset_group.rb +27 -0
  52. data/app/components/hakumi_components/concerns/form_field.rb +169 -68
  53. data/app/components/hakumi_components/concerns/form_field_contract.rb +57 -0
  54. data/app/components/hakumi_components/concerns/form_field_interface.rb +16 -0
  55. data/app/components/hakumi_components/concerns/input_control.rb +111 -0
  56. data/app/components/hakumi_components/concerns/input_control_contract.rb +47 -0
  57. data/app/components/hakumi_components/concerns/input_control_interface.rb +34 -0
  58. data/app/components/hakumi_components/concerns/selection_control.rb +122 -0
  59. data/app/components/hakumi_components/concerns/selection_control_contract.rb +47 -0
  60. data/app/components/hakumi_components/concerns/selection_control_interface.rb +34 -0
  61. data/app/components/hakumi_components/container/component.rb +29 -14
  62. data/app/components/hakumi_components/date_picker/component.html.erb +1 -3
  63. data/app/components/hakumi_components/date_picker/component.rb +130 -16
  64. data/app/components/hakumi_components/date_picker/range_picker.rb +159 -28
  65. data/app/components/hakumi_components/date_picker/shared_rendering.rb +151 -61
  66. data/app/components/hakumi_components/descriptions/component.html.erb +6 -6
  67. data/app/components/hakumi_components/descriptions/component.rb +98 -61
  68. data/app/components/hakumi_components/descriptions/item/component.rb +32 -8
  69. data/app/components/hakumi_components/divider/component.rb +36 -16
  70. data/app/components/hakumi_components/drawer/component.html.erb +2 -2
  71. data/app/components/hakumi_components/drawer/component.rb +112 -29
  72. data/app/components/hakumi_components/dropdown/component.rb +32 -35
  73. data/app/components/hakumi_components/dropdown/divider/component.rb +4 -0
  74. data/app/components/hakumi_components/dropdown/item/component.rb +42 -16
  75. data/app/components/hakumi_components/empty/component.rb +66 -31
  76. data/app/components/hakumi_components/flex/component.rb +39 -21
  77. data/app/components/hakumi_components/float_button/back_top/component.rb +35 -28
  78. data/app/components/hakumi_components/float_button/component.rb +161 -75
  79. data/app/components/hakumi_components/float_button/group/component.rb +85 -98
  80. data/app/components/hakumi_components/float_button/group_cluster/component.rb +46 -54
  81. data/app/components/hakumi_components/float_button/group_spec.rb +259 -0
  82. data/app/components/hakumi_components/float_button/item_spec.rb +115 -0
  83. data/app/components/hakumi_components/float_button/position_config.rb +68 -0
  84. data/app/components/hakumi_components/form/item/component.rb +34 -27
  85. data/app/components/hakumi_components/grid/col/component.rb +88 -17
  86. data/app/components/hakumi_components/grid/row/component.rb +35 -17
  87. data/app/components/hakumi_components/icon/component.rb +49 -49
  88. data/app/components/hakumi_components/image/component.html.erb +7 -7
  89. data/app/components/hakumi_components/image/component.rb +103 -40
  90. data/app/components/hakumi_components/image/preview_group/component.html.erb +1 -1
  91. data/app/components/hakumi_components/image/preview_group/component.rb +41 -36
  92. data/app/components/hakumi_components/image/preview_group/item.rb +48 -0
  93. data/app/components/hakumi_components/input/component.html.erb +1 -1
  94. data/app/components/hakumi_components/input/component.rb +142 -66
  95. data/app/components/hakumi_components/input/password/component.rb +30 -41
  96. data/app/components/hakumi_components/input/text_area/component.html.erb +2 -2
  97. data/app/components/hakumi_components/input/text_area/component.rb +89 -42
  98. data/app/components/hakumi_components/input_number/component.html.erb +5 -5
  99. data/app/components/hakumi_components/input_number/component.rb +137 -55
  100. data/app/components/hakumi_components/layout/component.html.erb +1 -1
  101. data/app/components/hakumi_components/layout/component.rb +23 -16
  102. data/app/components/hakumi_components/layout/content/component.html.erb +1 -1
  103. data/app/components/hakumi_components/layout/content/component.rb +24 -14
  104. data/app/components/hakumi_components/layout/footer/component.html.erb +1 -1
  105. data/app/components/hakumi_components/layout/footer/component.rb +24 -14
  106. data/app/components/hakumi_components/layout/header/component.html.erb +1 -1
  107. data/app/components/hakumi_components/layout/header/component.rb +24 -14
  108. data/app/components/hakumi_components/layout/sider/component.html.erb +1 -1
  109. data/app/components/hakumi_components/layout/sider/component.rb +83 -49
  110. data/app/components/hakumi_components/mentions/coercion.rb +46 -0
  111. data/app/components/hakumi_components/mentions/component.html.erb +6 -6
  112. data/app/components/hakumi_components/mentions/component.rb +131 -67
  113. data/app/components/hakumi_components/mentions/option.rb +30 -0
  114. data/app/components/hakumi_components/menu/component.rb +50 -59
  115. data/app/components/hakumi_components/menu/divider/component.rb +4 -0
  116. data/app/components/hakumi_components/menu/group/component.rb +16 -15
  117. data/app/components/hakumi_components/menu/item/component.rb +42 -19
  118. data/app/components/hakumi_components/menu/sub_menu/component.rb +49 -37
  119. data/app/components/hakumi_components/message/component.rb +62 -27
  120. data/app/components/hakumi_components/modal/component.html.erb +6 -6
  121. data/app/components/hakumi_components/modal/component.rb +93 -40
  122. data/app/components/hakumi_components/modal/confirm/component.html.erb +2 -2
  123. data/app/components/hakumi_components/modal/confirm/component.rb +49 -8
  124. data/app/components/hakumi_components/modal/error/component.rb +8 -39
  125. data/app/components/hakumi_components/modal/info/component.rb +6 -40
  126. data/app/components/hakumi_components/modal/status_component.rb +150 -0
  127. data/app/components/hakumi_components/modal/success/component.rb +6 -40
  128. data/app/components/hakumi_components/modal/warning/component.rb +14 -44
  129. data/app/components/hakumi_components/notification/component.rb +72 -38
  130. data/app/components/hakumi_components/pagination/component.html.erb +7 -11
  131. data/app/components/hakumi_components/pagination/component.rb +105 -72
  132. data/app/components/hakumi_components/pagination/page_item.rb +49 -0
  133. data/app/components/hakumi_components/popconfirm/component.rb +67 -19
  134. data/app/components/hakumi_components/popover/component.html.erb +2 -2
  135. data/app/components/hakumi_components/popover/component.rb +102 -38
  136. data/app/components/hakumi_components/progress/attribute_renderer.rb +354 -0
  137. data/app/components/hakumi_components/progress/circle_geometry.rb +76 -0
  138. data/app/components/hakumi_components/progress/component.html.erb +18 -17
  139. data/app/components/hakumi_components/progress/component.rb +230 -402
  140. data/app/components/hakumi_components/progress/controller_locals_parser.rb +130 -0
  141. data/app/components/hakumi_components/progress/info_tooltip_policy.rb +39 -0
  142. data/app/components/hakumi_components/progress/info_value.rb +75 -0
  143. data/app/components/hakumi_components/progress/status_state.rb +31 -0
  144. data/app/components/hakumi_components/progress/steps_renderer.rb +149 -0
  145. data/app/components/hakumi_components/progress/stroke_gradient_value.rb +90 -0
  146. data/app/components/hakumi_components/qr_code/component.rb +138 -72
  147. data/app/components/hakumi_components/radio/component.html.erb +3 -3
  148. data/app/components/hakumi_components/radio/component.rb +42 -9
  149. data/app/components/hakumi_components/radio/group/component.html.erb +9 -9
  150. data/app/components/hakumi_components/radio/group/component.rb +76 -27
  151. data/app/components/hakumi_components/radio/group/option.rb +43 -0
  152. data/app/components/hakumi_components/rate/component.html.erb +1 -1
  153. data/app/components/hakumi_components/rate/component.rb +55 -9
  154. data/app/components/hakumi_components/result/component.rb +87 -37
  155. data/app/components/hakumi_components/segmented/component.html.erb +1 -1
  156. data/app/components/hakumi_components/segmented/component.rb +81 -118
  157. data/app/components/hakumi_components/segmented/option.rb +191 -0
  158. data/app/components/hakumi_components/select/component.html.erb +19 -19
  159. data/app/components/hakumi_components/select/component.rb +88 -37
  160. data/app/components/hakumi_components/selection_control/coercion.rb +161 -0
  161. data/app/components/hakumi_components/selection_control/entry.rb +22 -0
  162. data/app/components/hakumi_components/selection_control/option.rb +42 -0
  163. data/app/components/hakumi_components/selection_control/option_group.rb +34 -0
  164. data/app/components/hakumi_components/selection_control/tree_node.rb +123 -0
  165. data/app/components/hakumi_components/skeleton/avatar/component.rb +36 -14
  166. data/app/components/hakumi_components/skeleton/avatar_config.rb +79 -0
  167. data/app/components/hakumi_components/skeleton/button/component.rb +32 -14
  168. data/app/components/hakumi_components/skeleton/component.rb +79 -93
  169. data/app/components/hakumi_components/skeleton/image/component.rb +46 -22
  170. data/app/components/hakumi_components/skeleton/input/component.rb +25 -10
  171. data/app/components/hakumi_components/skeleton/node/component.rb +23 -7
  172. data/app/components/hakumi_components/skeleton/paragraph_config.rb +92 -0
  173. data/app/components/hakumi_components/skeleton/title_config.rb +31 -0
  174. data/app/components/hakumi_components/slider/component.html.erb +11 -11
  175. data/app/components/hakumi_components/slider/component.rb +172 -53
  176. data/app/components/hakumi_components/slider/mark.rb +27 -0
  177. data/app/components/hakumi_components/space/compact/component.html.erb +1 -3
  178. data/app/components/hakumi_components/space/compact/component.rb +35 -19
  179. data/app/components/hakumi_components/space/component.html.erb +1 -1
  180. data/app/components/hakumi_components/space/component.rb +75 -39
  181. data/app/components/hakumi_components/spin/component.rb +92 -31
  182. data/app/components/hakumi_components/splitter/component.html.erb +9 -3
  183. data/app/components/hakumi_components/splitter/component.rb +55 -22
  184. data/app/components/hakumi_components/splitter/panel/component.html.erb +1 -3
  185. data/app/components/hakumi_components/splitter/panel/component.rb +64 -39
  186. data/app/components/hakumi_components/statistic/component.rb +143 -59
  187. data/app/components/hakumi_components/steps/component.rb +87 -55
  188. data/app/components/hakumi_components/steps/item/component.rb +59 -20
  189. data/app/components/hakumi_components/switch/component.html.erb +2 -3
  190. data/app/components/hakumi_components/switch/component.rb +62 -36
  191. data/app/components/hakumi_components/table/column/component.rb +80 -23
  192. data/app/components/hakumi_components/table/column_definition.rb +181 -0
  193. data/app/components/hakumi_components/table/column_group/component.rb +32 -21
  194. data/app/components/hakumi_components/table/component.html.erb +22 -31
  195. data/app/components/hakumi_components/table/component.rb +248 -509
  196. data/app/components/hakumi_components/table/concerns/attribute_helpers.rb +22 -0
  197. data/app/components/hakumi_components/table/concerns/cell_rendering.rb +277 -0
  198. data/app/components/hakumi_components/table/concerns/columns.rb +293 -91
  199. data/app/components/hakumi_components/table/concerns/editable.rb +36 -89
  200. data/app/components/hakumi_components/table/concerns/ellipsis.rb +31 -30
  201. data/app/components/hakumi_components/table/concerns/fixed_columns.rb +26 -19
  202. data/app/components/hakumi_components/table/concerns/surface_rendering.rb +365 -0
  203. data/app/components/hakumi_components/table/configs.rb +518 -0
  204. data/app/components/hakumi_components/table/definition_types.rb +40 -0
  205. data/app/components/hakumi_components/table/display_rows.rb +56 -0
  206. data/app/components/hakumi_components/table/ellipsis_config.rb +143 -0
  207. data/app/components/hakumi_components/table/fixed_offset.rb +36 -0
  208. data/app/components/hakumi_components/table/header_cell.rb +23 -0
  209. data/app/components/hakumi_components/table/row_record.rb +60 -0
  210. data/app/components/hakumi_components/table/row_render_state.rb +85 -0
  211. data/app/components/hakumi_components/tabs/component.html.erb +8 -8
  212. data/app/components/hakumi_components/tabs/component.rb +114 -64
  213. data/app/components/hakumi_components/tabs/indicator_config.rb +34 -0
  214. data/app/components/hakumi_components/tabs/item/component.rb +73 -13
  215. data/app/components/hakumi_components/tag/component.rb +142 -47
  216. data/app/components/hakumi_components/tag/group/component.rb +18 -6
  217. data/app/components/hakumi_components/time_picker/component.html.erb +8 -8
  218. data/app/components/hakumi_components/time_picker/component.rb +71 -17
  219. data/app/components/hakumi_components/timeline/component.html.erb +3 -2
  220. data/app/components/hakumi_components/timeline/component.rb +188 -55
  221. data/app/components/hakumi_components/timeline/item/component.html.erb +2 -2
  222. data/app/components/hakumi_components/timeline/item/component.rb +121 -60
  223. data/app/components/hakumi_components/tooltip/component.html.erb +1 -1
  224. data/app/components/hakumi_components/tooltip/component.rb +89 -64
  225. data/app/components/hakumi_components/tour/component.rb +74 -49
  226. data/app/components/hakumi_components/tour/step.rb +86 -0
  227. data/app/components/hakumi_components/transfer/coercion.rb +66 -0
  228. data/app/components/hakumi_components/transfer/component.html.erb +171 -140
  229. data/app/components/hakumi_components/transfer/component.rb +232 -82
  230. data/app/components/hakumi_components/transfer/item.rb +54 -0
  231. data/app/components/hakumi_components/transfer/operations.rb +31 -0
  232. data/app/components/hakumi_components/tree/component.rb +219 -400
  233. data/app/components/hakumi_components/tree/concerns/node_rendering.rb +365 -0
  234. data/app/components/hakumi_components/tree/node.rb +276 -0
  235. data/app/components/hakumi_components/tree_select/component.html.erb +23 -9
  236. data/app/components/hakumi_components/tree_select/component.rb +116 -107
  237. data/app/components/hakumi_components/typography/base_component.rb +69 -36
  238. data/app/components/hakumi_components/typography/link/component.rb +63 -6
  239. data/app/components/hakumi_components/typography/paragraph/component.rb +6 -0
  240. data/app/components/hakumi_components/typography/text/component.rb +4 -0
  241. data/app/components/hakumi_components/typography/title/component.rb +61 -6
  242. data/app/components/hakumi_components/upload/component.html.erb +106 -88
  243. data/app/components/hakumi_components/upload/component.rb +222 -48
  244. data/app/components/hakumi_components/upload/file_entry.rb +53 -0
  245. data/app/controllers/hakumi_components/components_controller.rb +2 -1
  246. data/app/form_builders/hakumi_components/form_builder/contracts.rb +55 -0
  247. data/app/form_builders/hakumi_components/form_builder/field_context.rb +43 -0
  248. data/app/form_builders/hakumi_components/form_builder/model_binding.rb +163 -0
  249. data/app/form_builders/hakumi_components/form_builder.rb +238 -317
  250. data/app/helpers/hakumi_components/form_helper.rb +27 -23
  251. data/app/javascript/hakumi_components/controllers/hakumi/admin_panel_controller.js +5 -7
  252. data/app/javascript/hakumi_components/controllers/hakumi/back_top_controller.js +1 -1
  253. data/app/javascript/hakumi_components/controllers/hakumi/button_controller.js +108 -2
  254. data/app/javascript/hakumi_components/controllers/hakumi/calendar_controller.js +183 -95
  255. data/app/javascript/hakumi_components/controllers/hakumi/color_picker_controller.js +23 -285
  256. data/app/javascript/hakumi_components/controllers/hakumi/date_picker_controller.js +274 -262
  257. data/app/javascript/hakumi_components/controllers/hakumi/float_button_group_controller.js +2 -2
  258. data/app/javascript/hakumi_components/controllers/hakumi/message_controller.js +4 -2
  259. data/app/javascript/hakumi_components/controllers/hakumi/modal_controller.js +119 -125
  260. data/app/javascript/hakumi_components/controllers/hakumi/table/editable.js +291 -0
  261. data/app/javascript/hakumi_components/controllers/hakumi/table_controller.js +166 -366
  262. data/app/javascript/hakumi_components/controllers/hakumi/tabs_controller.js +8 -2
  263. data/app/javascript/hakumi_components/controllers/hakumi/tag_controller.js +27 -25
  264. data/app/javascript/hakumi_components/controllers/hakumi/tag_group_controller.js +19 -18
  265. data/app/javascript/hakumi_components/controllers/hakumi/theme_controller.js +116 -117
  266. data/app/javascript/hakumi_components/controllers/hakumi/transfer_controller.js +17 -1
  267. data/app/javascript/hakumi_components/controllers/hakumi/tree_controller.js +363 -78
  268. data/app/javascript/hakumi_components/controllers/hakumi/typography_controller.js +3 -3
  269. data/app/javascript/hakumi_components/controllers/hakumi/upload_controller.js +320 -204
  270. data/app/javascript/hakumi_components/core/render_component.js +37 -11
  271. data/app/javascript/hakumi_components/utils/color_helper.js +262 -0
  272. data/app/javascript/stylesheets/_base.scss +9 -0
  273. data/app/javascript/stylesheets/_hakumi_components.scss +1 -0
  274. data/app/javascript/stylesheets/components/_breadcrumb.scss +2 -2
  275. data/app/javascript/stylesheets/components/_calendar.scss +13 -13
  276. data/app/javascript/stylesheets/components/_cascader.scss +5 -5
  277. data/app/javascript/stylesheets/components/_checkbox.scss +9 -11
  278. data/app/javascript/stylesheets/components/_color_picker.scss +11 -11
  279. data/app/javascript/stylesheets/components/_date_picker.scss +4 -4
  280. data/app/javascript/stylesheets/components/_descriptions.scss +2 -2
  281. data/app/javascript/stylesheets/components/_drawer.scss +3 -3
  282. data/app/javascript/stylesheets/components/_dropdown.scss +2 -2
  283. data/app/javascript/stylesheets/components/_flex.scss +1 -1
  284. data/app/javascript/stylesheets/components/_float_button.scss +5 -5
  285. data/app/javascript/stylesheets/components/_form_item.scss +92 -0
  286. data/app/javascript/stylesheets/components/_image.scss +15 -15
  287. data/app/javascript/stylesheets/components/_input.scss +23 -113
  288. data/app/javascript/stylesheets/components/_layout.scss +27 -26
  289. data/app/javascript/stylesheets/components/_menu.scss +15 -15
  290. data/app/javascript/stylesheets/components/_modal.scss +13 -13
  291. data/app/javascript/stylesheets/components/_notification.scss +3 -3
  292. data/app/javascript/stylesheets/components/_popover.scss +1 -1
  293. data/app/javascript/stylesheets/components/_segmented.scss +3 -3
  294. data/app/javascript/stylesheets/components/_select.scss +6 -6
  295. data/app/javascript/stylesheets/components/_slider.scss +1 -1
  296. data/app/javascript/stylesheets/components/_spin.scss +2 -2
  297. data/app/javascript/stylesheets/components/_steps.scss +10 -10
  298. data/app/javascript/stylesheets/components/_switch.scss +11 -10
  299. data/app/javascript/stylesheets/components/_table.scss +6 -6
  300. data/app/javascript/stylesheets/components/_tag.scss +2 -2
  301. data/app/javascript/stylesheets/components/_tooltip.scss +4 -4
  302. data/app/javascript/stylesheets/components/_tree_select.scss +3 -3
  303. data/app/javascript/stylesheets/components/_typography.scss +3 -3
  304. data/app/javascript/stylesheets/components/_upload.scss +4 -4
  305. data/app/services/hakumi_components/component_handler.rb +95 -24
  306. data/app/services/hakumi_components/icon/loader.rb +84 -67
  307. data/app/services/hakumi_components/illustrations/loader.rb +35 -27
  308. data/app/views/hakumi/_alert.html.erb +4 -1
  309. data/app/views/hakumi/_drawer.html.erb +1 -1
  310. data/app/views/hakumi/_statistic.html.erb +11 -11
  311. data/lib/generators/hakumi_components/install_generator.rb +14 -1
  312. data/lib/hakumi_components/documentation/models.rb +290 -0
  313. data/lib/hakumi_components/documentation/node.rb +73 -0
  314. data/lib/hakumi_components/documentation.rb +139 -62
  315. data/lib/hakumi_components/engine.rb +17 -11
  316. data/lib/hakumi_components/file_size_checker.rb +215 -0
  317. data/lib/hakumi_components/locales/en.yml +115 -0
  318. data/lib/hakumi_components/rails/attribute_introspection.rb +39 -107
  319. data/lib/hakumi_components/rails/model_reflection.rb +154 -0
  320. data/lib/hakumi_components/rails/validation_introspection.rb +25 -73
  321. data/lib/hakumi_components/rails/validation_mapper.rb +148 -201
  322. data/lib/hakumi_components/rails.rb +2 -0
  323. data/lib/hakumi_components/stylesheet_ownership_checker.rb +191 -0
  324. data/lib/hakumi_components/types/form_field.rb +24 -0
  325. data/lib/hakumi_components/types/html.rb +28 -0
  326. data/lib/hakumi_components/types/rendering.rb +10 -0
  327. data/lib/hakumi_components/types/stimulus.rb +19 -0
  328. data/lib/hakumi_components/types/validation.rb +22 -0
  329. data/lib/hakumi_components/types.rb +13 -0
  330. data/lib/hakumi_components/version.rb +2 -1
  331. data/lib/hakumi_components.rb +26 -6
  332. data/lib/tasks/coverage.rake +1 -0
  333. metadata +100 -136
  334. data/sig/action_view/tag_builder.rbs +0 -24
  335. data/sig/active_support/concern_patch.rbs +0 -7
  336. data/sig/generators/hakumi_components/install_generator.rbs +0 -11
  337. data/sig/hakumi_components/admin_panel/component.rbs +0 -28
  338. data/sig/hakumi_components/affix/component.rbs +0 -44
  339. data/sig/hakumi_components/alert/component.rbs +0 -64
  340. data/sig/hakumi_components/anchor/component.rbs +0 -45
  341. data/sig/hakumi_components/anchor/link/component.rbs +0 -31
  342. data/sig/hakumi_components/autocomplete/component.rbs +0 -68
  343. data/sig/hakumi_components/avatar/component.rbs +0 -44
  344. data/sig/hakumi_components/badge/component.rbs +0 -41
  345. data/sig/hakumi_components/base_component.rbs +0 -82
  346. data/sig/hakumi_components/breadcrumb/component.rbs +0 -21
  347. data/sig/hakumi_components/breadcrumb/item/component.rbs +0 -36
  348. data/sig/hakumi_components/button/component.rbs +0 -56
  349. data/sig/hakumi_components/calendar/component.rbs +0 -41
  350. data/sig/hakumi_components/card/component.rbs +0 -50
  351. data/sig/hakumi_components/card/grid/component.rbs +0 -21
  352. data/sig/hakumi_components/card/meta/component.rbs +0 -25
  353. data/sig/hakumi_components/carousel/component.rbs +0 -88
  354. data/sig/hakumi_components/cascader/component.rbs +0 -52
  355. data/sig/hakumi_components/checkbox/component.rbs +0 -46
  356. data/sig/hakumi_components/checkbox/group/component.rbs +0 -34
  357. data/sig/hakumi_components/collapse/component.rbs +0 -57
  358. data/sig/hakumi_components/collapse/panel/component.rbs +0 -47
  359. data/sig/hakumi_components/color_picker/component.rbs +0 -80
  360. data/sig/hakumi_components/concerns/form_field.rbs +0 -89
  361. data/sig/hakumi_components/container/component.rbs +0 -37
  362. data/sig/hakumi_components/date_picker/component.rbs +0 -72
  363. data/sig/hakumi_components/date_picker/range_picker.rbs +0 -68
  364. data/sig/hakumi_components/date_picker/shared_rendering.rbs +0 -41
  365. data/sig/hakumi_components/descriptions/component.rbs +0 -59
  366. data/sig/hakumi_components/descriptions/item/component.rbs +0 -31
  367. data/sig/hakumi_components/divider/component.rbs +0 -39
  368. data/sig/hakumi_components/drawer/component.rbs +0 -66
  369. data/sig/hakumi_components/dropdown/component.rbs +0 -45
  370. data/sig/hakumi_components/dropdown/divider/component.rbs +0 -11
  371. data/sig/hakumi_components/dropdown/item/component.rbs +0 -45
  372. data/sig/hakumi_components/empty/component.rbs +0 -40
  373. data/sig/hakumi_components/flex/component.rbs +0 -47
  374. data/sig/hakumi_components/float_button/back_top/component.rbs +0 -45
  375. data/sig/hakumi_components/float_button/component.rbs +0 -80
  376. data/sig/hakumi_components/float_button/group/component.rbs +0 -95
  377. data/sig/hakumi_components/float_button/group_cluster/component.rbs +0 -60
  378. data/sig/hakumi_components/form/item/component.rbs +0 -39
  379. data/sig/hakumi_components/form_builder.rbs +0 -37
  380. data/sig/hakumi_components/grid/col/component.rbs +0 -50
  381. data/sig/hakumi_components/grid/row/component.rbs +0 -46
  382. data/sig/hakumi_components/icon/component.rbs +0 -56
  383. data/sig/hakumi_components/image/component.rbs +0 -56
  384. data/sig/hakumi_components/image/preview_group/component.rbs +0 -28
  385. data/sig/hakumi_components/input/component.rbs +0 -71
  386. data/sig/hakumi_components/input/password/component.rbs +0 -27
  387. data/sig/hakumi_components/input/text_area/component.rbs +0 -42
  388. data/sig/hakumi_components/input_number/component.rbs +0 -85
  389. data/sig/hakumi_components/layout/component.rbs +0 -21
  390. data/sig/hakumi_components/layout/content/component.rbs +0 -19
  391. data/sig/hakumi_components/layout/footer/component.rbs +0 -19
  392. data/sig/hakumi_components/layout/header/component.rbs +0 -19
  393. data/sig/hakumi_components/layout/sider/component.rbs +0 -19
  394. data/sig/hakumi_components/mentions/component.rbs +0 -75
  395. data/sig/hakumi_components/menu/component.rbs +0 -52
  396. data/sig/hakumi_components/menu/divider/component.rbs +0 -11
  397. data/sig/hakumi_components/menu/group/component.rbs +0 -20
  398. data/sig/hakumi_components/menu/item/component.rbs +0 -42
  399. data/sig/hakumi_components/menu/sub_menu/component.rbs +0 -37
  400. data/sig/hakumi_components/message/component.rbs +0 -50
  401. data/sig/hakumi_components/modal/component.rbs +0 -76
  402. data/sig/hakumi_components/modal/confirm/component.rbs +0 -32
  403. data/sig/hakumi_components/modal/error/component.rbs +0 -27
  404. data/sig/hakumi_components/modal/info/component.rbs +0 -27
  405. data/sig/hakumi_components/modal/success/component.rbs +0 -27
  406. data/sig/hakumi_components/modal/warning/component.rbs +0 -27
  407. data/sig/hakumi_components/notification/component.rbs +0 -64
  408. data/sig/hakumi_components/pagination/component.rbs +0 -113
  409. data/sig/hakumi_components/popconfirm/component.rbs +0 -52
  410. data/sig/hakumi_components/popover/component.rbs +0 -66
  411. data/sig/hakumi_components/progress/component.rbs +0 -201
  412. data/sig/hakumi_components/qr_code/component.rbs +0 -111
  413. data/sig/hakumi_components/radio/component.rbs +0 -55
  414. data/sig/hakumi_components/radio/group/component.rbs +0 -71
  415. data/sig/hakumi_components/rails/attribute_introspection.rbs +0 -16
  416. data/sig/hakumi_components/rails/validation_introspection.rbs +0 -18
  417. data/sig/hakumi_components/rails/validation_mapper.rbs +0 -53
  418. data/sig/hakumi_components/rails.rbs +0 -6
  419. data/sig/hakumi_components/rate/component.rbs +0 -56
  420. data/sig/hakumi_components/result/component.rbs +0 -65
  421. data/sig/hakumi_components/segmented/component.rbs +0 -72
  422. data/sig/hakumi_components/select/component.rbs +0 -73
  423. data/sig/hakumi_components/skeleton/avatar/component.rbs +0 -23
  424. data/sig/hakumi_components/skeleton/button/component.rbs +0 -24
  425. data/sig/hakumi_components/skeleton/component.rbs +0 -76
  426. data/sig/hakumi_components/skeleton/image/component.rbs +0 -22
  427. data/sig/hakumi_components/skeleton/input/component.rbs +0 -23
  428. data/sig/hakumi_components/skeleton/node/component.rbs +0 -21
  429. data/sig/hakumi_components/slider/component.rbs +0 -85
  430. data/sig/hakumi_components/space/compact/component.rbs +0 -21
  431. data/sig/hakumi_components/space/component.rbs +0 -47
  432. data/sig/hakumi_components/spin/component.rbs +0 -77
  433. data/sig/hakumi_components/splitter/component.rbs +0 -27
  434. data/sig/hakumi_components/splitter/panel/component.rbs +0 -36
  435. data/sig/hakumi_components/statistic/component.rbs +0 -97
  436. data/sig/hakumi_components/steps/component.rbs +0 -82
  437. data/sig/hakumi_components/steps/item/component.rbs +0 -29
  438. data/sig/hakumi_components/switch/component.rbs +0 -52
  439. data/sig/hakumi_components/table/column/component.rbs +0 -57
  440. data/sig/hakumi_components/table/column_group/component.rbs +0 -20
  441. data/sig/hakumi_components/table/component.rbs +0 -279
  442. data/sig/hakumi_components/table/concerns/columns.rbs +0 -95
  443. data/sig/hakumi_components/table/concerns/editable.rbs +0 -40
  444. data/sig/hakumi_components/table/concerns/ellipsis.rbs +0 -27
  445. data/sig/hakumi_components/table/concerns/fixed_columns.rbs +0 -33
  446. data/sig/hakumi_components/tabs/component.rbs +0 -66
  447. data/sig/hakumi_components/tabs/item/component.rbs +0 -29
  448. data/sig/hakumi_components/tag/component.rbs +0 -86
  449. data/sig/hakumi_components/tag/group/component.rbs +0 -24
  450. data/sig/hakumi_components/time_picker/component.rbs +0 -72
  451. data/sig/hakumi_components/timeline/component.rbs +0 -61
  452. data/sig/hakumi_components/timeline/item/component.rbs +0 -70
  453. data/sig/hakumi_components/tooltip/component.rbs +0 -73
  454. data/sig/hakumi_components/tour/component.rbs +0 -84
  455. data/sig/hakumi_components/transfer/component.rbs +0 -75
  456. data/sig/hakumi_components/tree/component.rbs +0 -126
  457. data/sig/hakumi_components/tree_select/component.rbs +0 -122
  458. data/sig/hakumi_components/typography/base_component.rbs +0 -57
  459. data/sig/hakumi_components/typography/link/component.rbs +0 -28
  460. data/sig/hakumi_components/typography/paragraph/component.rbs +0 -13
  461. data/sig/hakumi_components/typography/text/component.rbs +0 -10
  462. data/sig/hakumi_components/typography/title/component.rbs +0 -28
  463. data/sig/hakumi_components/upload/component.rbs +0 -78
  464. data/sig/hakumi_components.rbs +0 -96
  465. data/sig/rails/active_model/validations/comparison_validator.rbs +0 -6
  466. data/sig/rails.rbs +0 -18
  467. data/sig/view_component/base.rbs +0 -28
@@ -1,74 +1,94 @@
1
- <div <%= tag.attributes(wrapper_attributes).to_s.html_safe %>>
2
- <div class="hakumi-auto-complete-input-wrapper">
3
- <% if custom_input_renderer? %>
4
- <%= input_renderer.call(input_attributes) %>
5
- <% else %>
6
- <input <%= tag.attributes(input_attributes).to_s.html_safe %>>
7
- <% end %>
8
-
9
- <% if allow_clear? %>
10
- <button
11
- type="button"
12
- class="hakumi-auto-complete-clear-icon"
13
- aria-label="Clear"
14
- data-hakumi--autocomplete-target="clear"
15
- data-action="click->hakumi--autocomplete#clear">
16
- <%= render HakumiComponents::Icon::Component.new(name: clear_icon, theme: :filled, fallback_to_outlined: true) %>
17
- </button>
18
- <% end %>
19
- </div>
1
+ <% autocomplete_control = capture do %>
2
+ <div <%= tag.attributes(wrapper_attributes).to_s.html_safe %>>
3
+ <div class="hakumi-auto-complete-input-wrapper">
4
+ <% if custom_input_renderer? %>
5
+ <%= input_renderer.call(input_attributes) %>
6
+ <% else %>
7
+ <input <%= tag.attributes(input_attributes).to_s.html_safe %>>
8
+ <% end %>
20
9
 
21
- <template data-hakumi--autocomplete-target="optionTemplate">
22
- <div
23
- class="hakumi-auto-complete-item"
24
- role="option"
25
- data-hakumi--autocomplete-target="option"
26
- data-action="mousedown->hakumi--autocomplete#select click->hakumi--autocomplete#select">
27
- <span class="hakumi-auto-complete-item-label"></span>
10
+ <% if allow_clear? %>
11
+ <button
12
+ type="button"
13
+ class="hakumi-auto-complete-clear-icon"
14
+ aria-label="Clear"
15
+ data-hakumi--autocomplete-target="clear"
16
+ data-action="click->hakumi--autocomplete#clear">
17
+ <%= render HakumiComponents::Icon::Component.new(name: clear_icon, theme: :filled, fallback_to_outlined: true) %>
18
+ </button>
19
+ <% end %>
28
20
  </div>
29
- </template>
30
21
 
31
- <div
32
- id="<%= dropdown_id %>"
33
- class="hakumi-auto-complete-dropdown hakumi-auto-complete-dropdown-hidden"
34
- role="listbox"
35
- data-hakumi--autocomplete-target="dropdown">
36
- <% if normalized_items.any? %>
37
- <% normalized_items.each do |item| %>
38
- <% if item[:type] == :group %>
39
- <div class="hakumi-auto-complete-group">
40
- <div class="hakumi-auto-complete-group-label"><%= item[:label] %></div>
41
- <% item[:options].each do |opt| %>
42
- <div
43
- class="hakumi-auto-complete-item"
44
- role="option"
45
- data-value="<%= opt[:value] %>"
46
- data-label="<%= opt[:label] %>"
47
- data-hakumi--autocomplete-target="option"
48
- data-action="mousedown->hakumi--autocomplete#select click->hakumi--autocomplete#select">
49
- <span class="hakumi-auto-complete-item-label"><%= opt[:label] %></span>
50
- </div>
51
- <% end %>
52
- </div>
53
- <% elsif item && item[:type] == :option %>
54
- <div
55
- class="hakumi-auto-complete-item"
56
- role="option"
57
- data-value="<%= item[:value] %>"
58
- data-label="<%= item[:label] %>"
59
- data-hakumi--autocomplete-target="option"
60
- data-action="mousedown->hakumi--autocomplete#select click->hakumi--autocomplete#select">
61
- <span class="hakumi-auto-complete-item-label"><%= item[:label] %></span>
62
- </div>
22
+ <template data-hakumi--autocomplete-target="optionTemplate">
23
+ <div
24
+ class="hakumi-auto-complete-item"
25
+ role="option"
26
+ data-hakumi--autocomplete-target="option"
27
+ data-action="mousedown->hakumi--autocomplete#select click->hakumi--autocomplete#select">
28
+ <span class="hakumi-auto-complete-item-label"></span>
29
+ </div>
30
+ </template>
31
+
32
+ <div
33
+ id="<%= dropdown_id %>"
34
+ class="hakumi-auto-complete-dropdown hakumi-auto-complete-dropdown-hidden"
35
+ role="listbox"
36
+ data-hakumi--autocomplete-target="dropdown">
37
+ <% if normalized_items.any? %>
38
+ <% normalized_items.each do |item| %>
39
+ <% if item.group? %>
40
+ <div class="hakumi-auto-complete-group">
41
+ <div class="hakumi-auto-complete-group-label"><%= item.label %></div>
42
+ <% item.options.each do |option| %>
43
+ <div
44
+ class="hakumi-auto-complete-item"
45
+ role="option"
46
+ data-value="<%= option.value %>"
47
+ data-label="<%= option.label %>"
48
+ data-hakumi--autocomplete-target="option"
49
+ data-action="mousedown->hakumi--autocomplete#select click->hakumi--autocomplete#select">
50
+ <span class="hakumi-auto-complete-item-label"><%= option.label %></span>
51
+ </div>
52
+ <% end %>
53
+ </div>
54
+ <% elsif item.option? %>
55
+ <div
56
+ class="hakumi-auto-complete-item"
57
+ role="option"
58
+ data-value="<%= item.value %>"
59
+ data-label="<%= item.label %>"
60
+ data-hakumi--autocomplete-target="option"
61
+ data-action="mousedown->hakumi--autocomplete#select click->hakumi--autocomplete#select">
62
+ <span class="hakumi-auto-complete-item-label"><%= item.label %></span>
63
+ </div>
64
+ <% end %>
63
65
  <% end %>
64
66
  <% end %>
65
- <% end %>
66
67
 
67
- <div
68
- class="hakumi-auto-complete-empty"
69
- data-hakumi--autocomplete-target="empty"
70
- style="display: <%= normalized_items.any? ? 'none' : 'block' %>;">
71
- <%= not_found_content %>
68
+ <div
69
+ class="hakumi-auto-complete-empty"
70
+ data-hakumi--autocomplete-target="empty"
71
+ style="display: <%= normalized_items.any? ? 'none' : 'block' %>;">
72
+ <%= not_found_content %>
73
+ </div>
74
+ </div>
75
+ </div>
76
+ <% end %>
77
+
78
+ <% if standalone? %>
79
+ <%= autocomplete_control %>
80
+ <% else %>
81
+ <div <%= tag.attributes(form_item_attributes).to_s.html_safe %>>
82
+ <%= render_label(field_id: input_id) %>
83
+
84
+ <div class="hakumi-form-item-control">
85
+ <div class="hakumi-form-item-control-input">
86
+ <div class="hakumi-form-item-control-input-content">
87
+ <%= autocomplete_control %>
88
+ </div>
89
+ </div>
90
+
91
+ <%= render_explain %>
72
92
  </div>
73
93
  </div>
74
- </div>
94
+ <% end %>
@@ -1,59 +1,115 @@
1
+ # typed: strict
1
2
  # frozen_string_literal: true
2
3
 
3
4
  module HakumiComponents
4
5
  module Autocomplete
5
- # Hakumi Autocomplete component
6
- # Text input with dropdown suggestions, built on top of Ant Design styles.
7
6
  class Component < HakumiComponents::BaseComponent
8
- SIZES = %i[small middle large].freeze
9
- VARIANTS = %i[outlined filled borderless underlined].freeze
10
- STATUSES = %i[error warning].freeze
11
- FILTER_MODES = %i[contains starts_with].freeze
12
-
13
- # @param name [String, Symbol] input name/id
14
- # @param options [Array] list of options or groups
15
- # - "label" (String) -> label & value are the same
16
- # - [label, value]
17
- # - { label:, value: }
18
- # - { label:, options: [...] } for grouped options
7
+ extend T::Sig
8
+
9
+ include HakumiComponents::Concerns::SelectionControl
10
+
11
+ InputRenderer = T.type_alias do
12
+ T.proc.params(attrs: Types::HtmlAttributes).returns(T.any(String, ActiveSupport::SafeBuffer))
13
+ end
14
+
15
+ VARIANTS = T.let(%i[outlined filled borderless underlined].freeze, T::Array[Symbol])
16
+ STATUSES = T.let(%i[error warning].freeze, T::Array[Symbol])
17
+ FILTER_MODES = T.let(%i[contains starts_with].freeze, T::Array[Symbol])
18
+
19
+ sig do
20
+ params(
21
+ name: Types::FormFieldName,
22
+ options: T.nilable(T::Array[HakumiComponents::SelectionControl::Coercion::EntryInput]),
23
+ label: T.nilable(String),
24
+ caption: T.nilable(String),
25
+ value: Types::FormFieldValue,
26
+ placeholder: T.nilable(String),
27
+ size: Symbol,
28
+ variant: Symbol,
29
+ status: T.nilable(Symbol),
30
+ disabled: T::Boolean,
31
+ required: T::Boolean,
32
+ allow_clear: T::Boolean,
33
+ clear_icon: Types::StringOrSymbol,
34
+ input_renderer: T.nilable(InputRenderer),
35
+ case_sensitive: T::Boolean,
36
+ filter_mode: Symbol,
37
+ not_found_content: T.nilable(String),
38
+ standalone: T::Boolean,
39
+ rules: T.nilable(T::Array[Types::ValidationRule]),
40
+ errors: Types::FormFieldErrors,
41
+ html_options: Types::HtmlAttributeValue
42
+ ).void
43
+ end
19
44
  def initialize(
20
45
  name:,
21
46
  options: [],
47
+ label: nil,
48
+ caption: nil,
22
49
  value: nil,
23
50
  placeholder: nil,
24
- size: :middle,
51
+ size: :default,
25
52
  variant: :outlined,
26
53
  status: nil,
27
54
  disabled: false,
55
+ required: false,
28
56
  allow_clear: false,
29
57
  clear_icon: :close_circle,
30
58
  input_renderer: nil,
31
59
  case_sensitive: false,
32
60
  filter_mode: :contains,
33
- not_found_content: "No options",
61
+ not_found_content: nil,
62
+ standalone: true,
63
+ rules: nil,
64
+ errors: [],
34
65
  **html_options
35
66
  )
36
- @name = name
37
- @options = options
38
- @value = value
67
+ @name = T.let(name, Types::FormFieldName)
68
+ @options = T.let(HakumiComponents::SelectionControl::Coercion.entries(Array(options)), T::Array[HakumiComponents::SelectionControl::Entry])
69
+ @label = label
70
+ @caption = caption
71
+ @value = T.let(value, Types::FormFieldValue)
39
72
  @placeholder = placeholder
40
73
  @size = size
41
74
  @variant = variant
42
75
  @status = status
43
76
  @disabled = disabled
77
+ @required = required
44
78
  @allow_clear = allow_clear
45
79
  @clear_icon = clear_icon
46
- @input_renderer = input_renderer
80
+ @input_renderer = T.let(input_renderer, T.nilable(InputRenderer))
47
81
  @case_sensitive = case_sensitive
48
82
  @filter_mode = filter_mode
49
83
  @not_found_content = not_found_content
50
- @html_options = html_options
84
+ @standalone = standalone
85
+ @rules = T.let(rules, T.nilable(T::Array[Types::ValidationRule]))
86
+ @errors = T.let(errors, Types::FormFieldErrors)
87
+ @html_options = T.let(html_options, Types::HtmlAttributes)
88
+ initialize_selection_control_contract!(
89
+ name: @name,
90
+ label: @label,
91
+ caption: @caption,
92
+ value: @value,
93
+ placeholder: @placeholder,
94
+ size: @size,
95
+ disabled: @disabled,
96
+ allow_clear: @allow_clear,
97
+ required: @required,
98
+ standalone: @standalone,
99
+ errors: @errors,
100
+ html_options: @html_options,
101
+ rules: @rules
102
+ )
51
103
 
52
104
  validate_props!
53
105
  end
54
106
 
55
107
  private
56
108
 
109
+ sig { returns(T::Array[HakumiComponents::SelectionControl::Entry]) }
110
+ attr_reader :options
111
+
112
+ sig { void }
57
113
  def validate_props!
58
114
  validate_inclusion!(:size, SIZES)
59
115
  validate_inclusion!(:variant, VARIANTS)
@@ -61,141 +117,111 @@ module HakumiComponents
61
117
  validate_inclusion!(:status, STATUSES) if @status
62
118
  end
63
119
 
120
+ sig { returns(String) }
64
121
  def wrapper_classes
65
122
  class_names(
66
123
  "auto-complete",
67
124
  {
68
- @size => (@size != :middle),
125
+ selection_control_size => selection_control_size != :default,
69
126
  "status-error": @status == :error,
70
127
  "status-warning": @status == :warning,
71
128
  "varihakumi-outlined": @variant == :outlined,
72
129
  "varihakumi-filled": @variant == :filled,
73
130
  "varihakumi-borderless": @variant == :borderless,
74
131
  "varihakumi-underlined": @variant == :underlined,
75
- disabled: @disabled
132
+ disabled: selection_control_disabled
76
133
  },
77
- [ @html_options[:class] ]
134
+ [ selection_control_html_class_name ]
78
135
  )
79
136
  end
80
137
 
138
+ sig { returns(Types::HtmlAttributes) }
81
139
  def wrapper_attributes
82
140
  base_attrs = {
83
141
  class: wrapper_classes,
84
142
  data: {
85
143
  controller: "hakumi--autocomplete",
86
- "hakumi--autocomplete-case-sensitive-value": @case_sensitive ? "true" : "false",
87
- "hakumi--autocomplete-filter-mode-value": @filter_mode.to_s
144
+ "hakumi--autocomplete-case-sensitive-value": @case_sensitive,
145
+ "hakumi--autocomplete-filter-mode-value": @filter_mode.to_s,
146
+ "hakumi--autocomplete-disabled-value": selection_control_disabled
88
147
  }
89
148
  }
90
149
 
91
- extra = @html_options.except(:class)
92
- merge_attributes(base_attrs, extra)
150
+ merge_attributes(base_attrs, selection_control_html_options_without_class)
93
151
  end
94
152
 
153
+ sig { returns(Types::HtmlAttributes) }
95
154
  def input_attributes
96
155
  attrs = {
97
156
  type: "text",
98
- name: @name,
99
- id: @html_options[:id] || @name,
100
- value: @value,
101
- placeholder: @placeholder,
102
- disabled: (@disabled ? true : nil),
157
+ name: form_field_name,
158
+ id: input_id,
159
+ value: selection_control_value,
160
+ placeholder: selection_control_placeholder,
161
+ disabled: selection_control_disabled ? true : nil,
103
162
  autocomplete: "off",
104
163
  spellcheck: "false",
105
- aria: {
106
- autocomplete: "list",
107
- controls: dropdown_id,
108
- expanded: "false"
109
- },
164
+ "aria-autocomplete": "list",
165
+ "aria-controls": dropdown_id,
166
+ "aria-expanded": "false",
167
+ "aria-haspopup": "listbox",
168
+ "aria-invalid": error? ? "true" : nil,
169
+ "aria-describedby": describedby_ids,
110
170
  data: {
111
171
  action: "input->hakumi--autocomplete#onInput keydown->hakumi--autocomplete#onKeydown focus->hakumi--autocomplete#open blur->hakumi--autocomplete#onBlur",
112
172
  "hakumi--autocomplete-target": "input"
113
173
  }
114
- }
174
+ }.compact
115
175
 
116
176
  attrs[:class] = input_classes unless custom_input_renderer?
117
- attrs[:aria].compact!
118
177
  attrs
119
178
  end
120
179
 
180
+ sig { returns(String) }
121
181
  def input_classes
122
- classes = [ "hakumi-input" ]
123
- classes << "hakumi-input-#{@size}" if @size != :middle
182
+ classes = T.let([ "hakumi-input" ], T::Array[String])
183
+ classes << "hakumi-input-#{selection_control_size}" if selection_control_size != :default
124
184
  classes.join(" ")
125
185
  end
126
186
 
187
+ sig { returns(String) }
127
188
  def dropdown_id
128
- @dropdown_id ||= "#{@name}_autocomplete_dropdown"
189
+ "#{input_id}_autocomplete_dropdown"
129
190
  end
130
191
 
192
+ sig { returns(T::Boolean) }
131
193
  def allow_clear?
132
- @allow_clear
194
+ selection_control_allow_clear
133
195
  end
134
196
 
197
+ sig { returns(T::Boolean) }
135
198
  def custom_input_renderer?
136
- @input_renderer.respond_to?(:call)
199
+ !@input_renderer.nil?
137
200
  end
138
201
 
202
+ sig { returns(T.nilable(InputRenderer)) }
139
203
  def input_renderer
140
204
  @input_renderer
141
205
  end
142
206
 
207
+ sig { returns(T::Boolean) }
143
208
  def disabled?
144
- @disabled
209
+ selection_control_disabled
145
210
  end
146
211
 
212
+ sig { returns(Types::StringOrSymbol) }
147
213
  def clear_icon
148
214
  @clear_icon
149
215
  end
150
216
 
217
+ sig { returns(String) }
151
218
  def not_found_content
152
- @not_found_content
219
+ @not_found_content || t_default(:not_found, default: "No options")
153
220
  end
154
221
 
222
+ sig { returns(T::Array[HakumiComponents::SelectionControl::Entry]) }
155
223
  def normalized_items
156
- @normalized_items ||= begin
157
- Array(@options).flat_map do |opt|
158
- if opt.is_a?(Hash) && opt[:options]
159
- {
160
- type: :group,
161
- label: opt[:label].to_s,
162
- options: normalize_group_options(opt[:options])
163
- }
164
- else
165
- normalize_simple_option(opt)
166
- end
167
- end
168
- end
169
- end
170
-
171
- def normalize_group_options(options)
172
- Array(options).map { |opt| normalize_simple_option(opt) }.compact
173
- end
174
-
175
- def normalize_simple_option(opt)
176
- case opt
177
- when nil
178
- nil
179
- when Hash
180
- {
181
- type: :option,
182
- label: opt[:label].to_s,
183
- value: (opt[:value] || opt[:label]).to_s
184
- }
185
- when Array
186
- label, value = opt
187
- {
188
- type: :option,
189
- label: label.to_s,
190
- value: (value || label).to_s
191
- }
192
- else
193
- {
194
- type: :option,
195
- label: opt.to_s,
196
- value: opt.to_s
197
- }
198
- end
224
+ options
199
225
  end
200
226
  end
201
227
  end
@@ -1,11 +1,23 @@
1
+ # typed: strict
1
2
  # frozen_string_literal: true
2
3
 
3
4
  module HakumiComponents
4
5
  module Avatar
5
6
  class Component < HakumiComponents::BaseComponent
6
- SHAPES = [ :circle, :square ].freeze
7
- SIZES = [ :small, :default, :large ].freeze
7
+ extend T::Sig
8
8
 
9
+ SHAPES = T.let([ :circle, :square ].freeze, T::Array[Symbol])
10
+
11
+ sig do
12
+ params(
13
+ shape: Symbol,
14
+ size: T.any(Symbol, Integer),
15
+ src: T.nilable(String),
16
+ icon: T.nilable(Types::StringOrSymbol),
17
+ alt: T.nilable(String),
18
+ html_options: Types::HtmlAttributeValue
19
+ ).void
20
+ end
9
21
  def initialize(
10
22
  shape: :circle,
11
23
  size: :default,
@@ -17,24 +29,27 @@ module HakumiComponents
17
29
  @shape = shape
18
30
  @size = size
19
31
  @src = src
20
- @icon = icon
32
+ @icon = T.let(icon, T.nilable(Types::StringOrSymbol))
21
33
  @alt = alt
22
- @html_options = html_options
34
+ @html_options = T.let(html_options, Types::HtmlAttributes)
23
35
 
24
36
  validate_props!
25
37
  end
26
38
 
39
+ sig { returns(ActiveSupport::SafeBuffer) }
27
40
  def call
28
41
  content_tag(:span, content_wrapper, wrapper_attributes)
29
42
  end
30
43
 
31
44
  private
32
45
 
46
+ sig { void }
33
47
  def validate_props!
34
48
  validate_inclusion!(:shape, SHAPES)
35
49
  validate_inclusion!(:size, SIZES) unless @size.is_a?(Integer)
36
50
  end
37
51
 
52
+ sig { returns(String) }
38
53
  def wrapper_classes
39
54
  class_names(
40
55
  "avatar",
@@ -42,24 +57,25 @@ module HakumiComponents
42
57
  @shape => true,
43
58
  lg: @size == :large,
44
59
  sm: @size == :small,
45
- icon: @icon.present?,
60
+ icon: @icon.present? && @src.blank?,
46
61
  image: @src.present?
47
62
  },
48
- [ @html_options[:class] ]
63
+ [ html_class_name ]
49
64
  )
50
65
  end
51
66
 
67
+ sig { returns(ActiveSupport::SafeBuffer) }
52
68
  def content_wrapper
53
69
  if @src
54
70
  image_tag(@src, alt: @alt, width: avatar_size, height: avatar_size)
55
71
  elsif @icon
56
- # Pass size: nil to allow icon to inherit font-size from avatar
57
72
  render HakumiComponents::Icon::Component.new(name: @icon, size: nil)
58
73
  else
59
74
  tag.span(content, data: { hakumi__avatar_text_target: "text" })
60
75
  end
61
76
  end
62
77
 
78
+ sig { returns(Integer) }
63
79
  def avatar_size
64
80
  case @size
65
81
  when :large then 40
@@ -69,6 +85,7 @@ module HakumiComponents
69
85
  end
70
86
  end
71
87
 
88
+ sig { returns(Types::HtmlAttributes) }
72
89
  def wrapper_attributes
73
90
  attrs = {
74
91
  class: wrapper_classes
@@ -78,13 +95,29 @@ module HakumiComponents
78
95
  attrs[:style] = "width: #{@size}px; height: #{@size}px; font-size: #{@size / 2}px;"
79
96
  end
80
97
 
81
- # Add controller for text avatars
82
- unless @src || @icon
83
- attrs[:data] ||= {}
84
- attrs[:data][:controller] = "hakumi--avatar-text"
98
+ if text_avatar?
99
+ attrs[:data] = text_avatar_data
85
100
  end
86
101
 
87
- merge_attributes(attrs, @html_options.except(:class))
102
+ merge_attributes(attrs, @html_options.except(:class, :data))
103
+ end
104
+
105
+ sig { returns(T::Boolean) }
106
+ def text_avatar?
107
+ @src.blank? && @icon.blank?
108
+ end
109
+
110
+ sig { returns(Types::DataAttributes) }
111
+ def text_avatar_data
112
+ data = data_attributes_from(@html_options[:data])
113
+ data[:controller] = append_data_token(data[:controller], "hakumi--avatar-text")
114
+ data
115
+ end
116
+
117
+ sig { returns(T.nilable(String)) }
118
+ def html_class_name
119
+ css_class = @html_options[:class]
120
+ css_class.is_a?(String) ? css_class : nil
88
121
  end
89
122
  end
90
123
  end