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
@@ -0,0 +1,130 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Progress
6
+ class ControllerLocalsParser
7
+ extend T::Sig
8
+
9
+ DisplaySize = T.type_alias { T.nilable(BaseComponent::SizeValue) }
10
+ StrokeGradient = T.type_alias { T::Hash[String, String] }
11
+ StrokeColor = T.type_alias { T.nilable(T.any(String, StrokeGradient)) }
12
+ TooltipOptionsMap = T.type_alias { T::Hash[Types::HtmlKey, Types::HtmlPrimitive] }
13
+
14
+ class << self
15
+ extend T::Sig
16
+
17
+ sig { params(params: BaseComponent::ControllerOptions).returns(Types::HtmlAttributes) }
18
+ def call(params)
19
+ {
20
+ percent: parse_number(html_param(params, :percent)),
21
+ success_percent: parse_number(html_param(params, :success_percent)),
22
+ type: symbol_html_param(html_param(params, :type)),
23
+ status: symbol_html_param(html_param(params, :status)),
24
+ show_info: boolean_html_param(html_param(params, :show_info)),
25
+ size: parse_size(html_param(params, :size)),
26
+ stroke_width: parse_number(html_param(params, :stroke_width)),
27
+ stroke_color: parse_stroke_color(html_param(params, :stroke_color)),
28
+ trail_color: string_html_param(html_param(params, :trail_color)),
29
+ stroke_linecap: symbol_html_param(html_param(params, :stroke_linecap)),
30
+ steps: integer_html_param(html_param(params, :steps)),
31
+ gap_degree: parse_number(html_param(params, :gap_degree)),
32
+ gap_position: symbol_html_param(html_param(params, :gap_position)),
33
+ format: string_html_param(html_param(params, :format)),
34
+ info_tooltip: html_primitive_param(html_param(params, :info_tooltip)),
35
+ info_tooltip_threshold: parse_number(html_param(params, :info_tooltip_threshold)),
36
+ info_tooltip_options: parse_tooltip_options(html_param(params, :info_tooltip_options))
37
+ }.compact
38
+ end
39
+
40
+ private
41
+
42
+ sig { params(params: BaseComponent::ControllerOptions, key: Symbol).returns(Types::HtmlAttributeValue) }
43
+ def html_param(params, key)
44
+ BaseComponent.html_param(params, key)
45
+ end
46
+
47
+ sig { params(value: Types::HtmlAttributeValue).returns(Types::HtmlPrimitive) }
48
+ def html_primitive_param(value)
49
+ BaseComponent.html_primitive_param(value)
50
+ end
51
+
52
+ sig { params(value: Types::HtmlAttributeValue).returns(T.nilable(T::Boolean)) }
53
+ def boolean_html_param(value)
54
+ BaseComponent.boolean_html_param(value)
55
+ end
56
+
57
+ sig { params(value: Types::HtmlAttributeValue).returns(T.nilable(Symbol)) }
58
+ def symbol_html_param(value)
59
+ BaseComponent.symbol_html_param(value)
60
+ end
61
+
62
+ sig { params(value: Types::HtmlAttributeValue).returns(T.nilable(Integer)) }
63
+ def integer_html_param(value)
64
+ BaseComponent.integer_html_param(value)
65
+ end
66
+
67
+ sig { params(value: Types::HtmlAttributeValue).returns(T.nilable(String)) }
68
+ def string_html_param(value)
69
+ BaseComponent.string_html_param(value)
70
+ end
71
+
72
+ sig { params(value: Types::HtmlAttributeValue).returns(T.nilable(Float)) }
73
+ def parse_number(value)
74
+ return nil if value.nil?
75
+ return value.to_f if value.is_a?(Numeric)
76
+ return nil unless value.is_a?(String)
77
+
78
+ stripped = value.strip
79
+ return nil if stripped.empty?
80
+
81
+ Float(stripped)
82
+ rescue ArgumentError
83
+ nil
84
+ end
85
+
86
+ sig { params(value: Types::HtmlAttributeValue).returns(DisplaySize) }
87
+ def parse_size(value)
88
+ return nil if value.nil?
89
+ return value if value.is_a?(Integer)
90
+ return value.to_i if value.is_a?(Float)
91
+
92
+ if value.is_a?(String)
93
+ stripped = value.strip
94
+ return nil if stripped.empty?
95
+
96
+ size_name = stripped.to_sym
97
+ return size_name if Component::LINE_SIZES.include?(size_name)
98
+
99
+ return stripped.to_i
100
+ end
101
+
102
+ nil
103
+ end
104
+
105
+ sig { params(value: Types::HtmlAttributeValue).returns(StrokeColor) }
106
+ def parse_stroke_color(value)
107
+ return nil if value.nil?
108
+ return value if value.is_a?(String) && value.present?
109
+ return value.to_s if value.is_a?(Symbol)
110
+
111
+ return StrokeGradientValue.normalize(value, compact_blank: false) if value.is_a?(Hash)
112
+
113
+ nil
114
+ end
115
+
116
+ sig { params(value: Types::HtmlAttributeValue).returns(T.nilable(TooltipOptionsMap)) }
117
+ def parse_tooltip_options(value)
118
+ return nil unless value.is_a?(Hash)
119
+
120
+ options = T.let({}, TooltipOptionsMap)
121
+ value.each do |key, entry_value|
122
+ primitive = html_primitive_param(entry_value)
123
+ options[key] = primitive
124
+ end
125
+ options
126
+ end
127
+ end
128
+ end
129
+ end
130
+ end
@@ -0,0 +1,39 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Progress
6
+ class InfoTooltipPolicy
7
+ extend T::Sig
8
+
9
+ TooltipMode = T.type_alias { T.nilable(Symbol) }
10
+
11
+ sig { params(mode: TooltipMode, type: Symbol, circle_size: Integer, threshold: Integer).void }
12
+ def initialize(mode:, type:, circle_size:, threshold:)
13
+ @mode = T.let(mode, TooltipMode)
14
+ @type = T.let(type, Symbol)
15
+ @circle_size = T.let(circle_size, Integer)
16
+ @threshold = T.let(threshold, Integer)
17
+ end
18
+
19
+ sig { returns(T::Boolean) }
20
+ def enabled?
21
+ !@mode.nil?
22
+ end
23
+
24
+ sig { returns(T::Boolean) }
25
+ def wrap?
26
+ return false unless enabled?
27
+
28
+ case @mode
29
+ when :always
30
+ true
31
+ when :auto
32
+ @type == :circle && @circle_size <= @threshold
33
+ else
34
+ false
35
+ end
36
+ end
37
+ end
38
+ end
39
+ end
@@ -0,0 +1,75 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Progress
6
+ class InfoValue
7
+ extend T::Sig
8
+
9
+ ProgressValue = T.type_alias { Numeric }
10
+ FormatProc = T.type_alias do
11
+ T.proc.params(
12
+ percent: ProgressValue,
13
+ success_percent: T.nilable(ProgressValue)
14
+ ).returns(String)
15
+ end
16
+ FormatValue = T.type_alias { T.nilable(T.any(Types::Renderable, FormatProc)) }
17
+
18
+ sig do
19
+ params(
20
+ percent: ProgressValue,
21
+ success_percent: T.nilable(ProgressValue),
22
+ content: T.untyped,
23
+ format_value: FormatValue,
24
+ rendered_format: T.untyped
25
+ ).void
26
+ end
27
+ def initialize(percent:, success_percent:, content:, format_value:, rendered_format:)
28
+ @percent = T.let(percent, ProgressValue)
29
+ @success_percent = T.let(success_percent, T.nilable(ProgressValue))
30
+ @content = T.let(content, T.untyped)
31
+ @format_value = T.let(format_value, FormatValue)
32
+ @rendered_format = T.let(rendered_format, T.untyped)
33
+ end
34
+
35
+ sig { returns(T.untyped) }
36
+ def text
37
+ return @content if present?(@content)
38
+
39
+ format_value = @format_value
40
+ return format_value.call(@percent, @success_percent) if format_value.is_a?(Proc)
41
+ return @rendered_format if present?(@rendered_format)
42
+
43
+ percent_label
44
+ end
45
+
46
+ sig { returns(String) }
47
+ def tooltip_label
48
+ value = text
49
+ present?(value) ? value.to_s : percent_label
50
+ end
51
+
52
+ sig { returns(T::Boolean) }
53
+ def custom?
54
+ present?(@content) || !@format_value.nil?
55
+ end
56
+
57
+ sig { returns(String) }
58
+ def percent_label
59
+ whole_number?(@percent.to_f) ? "#{@percent.to_i}%" : "#{@percent}%"
60
+ end
61
+
62
+ private
63
+
64
+ sig { params(value: T.untyped).returns(T::Boolean) }
65
+ def present?(value)
66
+ value.respond_to?(:present?) ? value.present? : !value.nil?
67
+ end
68
+
69
+ sig { params(value: ProgressValue).returns(T::Boolean) }
70
+ def whole_number?(value)
71
+ (value.to_f % 1.0).zero?
72
+ end
73
+ end
74
+ end
75
+ end
@@ -0,0 +1,31 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Progress
6
+ class StatusState
7
+ extend T::Sig
8
+
9
+ ProgressValue = T.type_alias { Numeric }
10
+
11
+ sig { params(percent: ProgressValue, explicit_status: T.nilable(Symbol)).void }
12
+ def initialize(percent:, explicit_status:)
13
+ @percent = T.let(percent, ProgressValue)
14
+ @explicit_status = T.let(explicit_status, T.nilable(Symbol))
15
+ end
16
+
17
+ sig { returns(Symbol) }
18
+ def value
19
+ status = @explicit_status
20
+ return status unless status.nil?
21
+
22
+ @percent.to_f >= 100 ? :success : :normal
23
+ end
24
+
25
+ sig { returns(T::Boolean) }
26
+ def explicit?
27
+ !@explicit_status.nil?
28
+ end
29
+ end
30
+ end
31
+ end
@@ -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