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,123 +1,187 @@
1
+ # typed: strict
1
2
  # frozen_string_literal: true
2
3
 
3
4
  module HakumiComponents
4
5
  module Popover
5
6
  class Component < HakumiComponents::BaseComponent
6
- PLACEMENTS = %w[
7
+ extend T::Sig
8
+
9
+ Placement = T.type_alias { T.any(Symbol, String) }
10
+ TriggerValue = T.type_alias { T.any(Symbol, String, T::Array[T.any(Symbol, String)]) }
11
+
12
+ PLACEMENTS = T.let(%i[
7
13
  top topLeft topRight
8
14
  bottom bottomLeft bottomRight
9
15
  left leftTop leftBottom
10
16
  right rightTop rightBottom
11
17
  auto
12
- ].freeze
18
+ ].freeze, T::Array[Symbol])
13
19
 
14
- TRIGGERS = %w[click hover focus].freeze
20
+ TRIGGERS = T.let(%i[click hover focus].freeze, T::Array[Symbol])
15
21
 
22
+ sig do
23
+ params(
24
+ params: T.any(ActionController::Parameters, HakumiComponents::Types::HtmlAttributes)
25
+ ).returns(HakumiComponents::Types::HtmlAttributes)
26
+ end
16
27
  def self.extract_controller_locals(params)
17
28
  {
18
- title: params[:title],
19
- body: params[:body],
20
- trigger_body: params[:trigger_body],
21
- placement: params[:placement],
22
- trigger: params[:trigger],
23
- open: ActiveModel::Type::Boolean.new.cast(params[:open]),
24
- arrow: params[:arrow].nil? ? true : ActiveModel::Type::Boolean.new.cast(params[:arrow]),
25
- disabled: ActiveModel::Type::Boolean.new.cast(params[:disabled]),
26
- auto_adjust_overflow: params[:auto_adjust_overflow].nil? ? true : ActiveModel::Type::Boolean.new.cast(params[:auto_adjust_overflow])
29
+ title: html_primitive_param(html_param(params, :title)),
30
+ body: html_primitive_param(html_param(params, :body)),
31
+ trigger_body: html_primitive_param(html_param(params, :trigger_body)),
32
+ placement: symbol_html_param(html_param(params, :placement)),
33
+ trigger: trigger_param(html_param(params, :trigger)),
34
+ open: boolean_html_param(html_param(params, :open)),
35
+ arrow: html_param(params, :arrow).nil? ? true : boolean_html_param(html_param(params, :arrow)),
36
+ disabled: boolean_html_param(html_param(params, :disabled)),
37
+ auto_adjust_overflow: html_param(params, :auto_adjust_overflow).nil? ? true : boolean_html_param(html_param(params, :auto_adjust_overflow))
27
38
  }
28
39
  end
29
40
 
41
+ sig do
42
+ params(
43
+ title: HakumiComponents::Types::Renderable,
44
+ body: HakumiComponents::Types::Renderable,
45
+ trigger_body: HakumiComponents::Types::Renderable,
46
+ placement: Placement,
47
+ trigger: T.nilable(TriggerValue),
48
+ open: T.nilable(T::Boolean),
49
+ arrow: T::Boolean,
50
+ disabled: T::Boolean,
51
+ auto_adjust_overflow: T::Boolean,
52
+ html_options: HakumiComponents::Types::HtmlAttributeValue
53
+ ).void
54
+ end
30
55
  def initialize(
31
56
  title: nil,
32
57
  body: nil,
33
58
  trigger_body: nil,
34
- placement: "top",
35
- trigger: "hover",
59
+ placement: :top,
60
+ trigger: :hover,
36
61
  open: nil,
37
62
  arrow: true,
38
63
  disabled: false,
39
64
  auto_adjust_overflow: true,
40
65
  **html_options
41
66
  )
42
- @title = title
43
- @body = body
44
- @trigger_body = trigger_body
45
- @placement = placement
46
- @trigger = trigger
47
- @open = open
48
- @arrow = arrow
49
- @disabled = disabled
50
- @auto_adjust_overflow = auto_adjust_overflow
51
- @html_options = html_options
52
- @wrapper_id = html_options[:id] || html_options["id"]
67
+ @title = T.let(title, HakumiComponents::Types::Renderable)
68
+ @body = T.let(body, HakumiComponents::Types::Renderable)
69
+ @trigger_body = T.let(trigger_body, HakumiComponents::Types::Renderable)
70
+ @placement = T.let(placement.to_sym, Symbol)
71
+ @trigger = T.let(trigger, T.nilable(TriggerValue))
72
+ @open = T.let(open, T.nilable(T::Boolean))
73
+ @arrow = T.let(arrow, T::Boolean)
74
+ @disabled = T.let(disabled, T::Boolean)
75
+ @auto_adjust_overflow = T.let(auto_adjust_overflow, T::Boolean)
76
+ @html_options = T.let(html_options, HakumiComponents::Types::HtmlAttributes)
77
+ raw_wrapper_id = @html_options[:id]
78
+ @wrapper_id = T.let(raw_wrapper_id&.to_s, T.nilable(String))
53
79
 
54
80
  validate_props!
55
81
  end
56
82
 
83
+ sig { returns(String) }
57
84
  def wrapper_classes
58
- classes = [ "hakumi-popover-wrapper" ]
59
- classes << "hakumi-popover-disabled" if @disabled
60
- classes << @html_options[:class] if @html_options[:class]
61
- classes.compact.join(" ")
85
+ extras = T.let([], T::Array[T.nilable(String)])
86
+ extras << "hakumi-popover-disabled" if @disabled
87
+ extras.concat(Array(html_classes(@html_options)).compact)
88
+
89
+ class_names(
90
+ "popover-wrapper",
91
+ {},
92
+ extras
93
+ )
62
94
  end
63
95
 
96
+ sig { returns(T.nilable(String)) }
64
97
  def wrapper_id
65
98
  @wrapper_id
66
99
  end
67
100
 
101
+ sig { returns(T.nilable(String)) }
68
102
  def popover_dom_id
69
103
  return if wrapper_id.blank?
70
104
  "#{wrapper_id}-popover"
71
105
  end
72
106
 
107
+ sig { returns(String) }
73
108
  def popover_classes
74
- classes = [ "hakumi-popover" ]
75
- classes << "hakumi-popover-arrow-hidden" unless @arrow
76
- classes.join(" ")
109
+ class_names("popover", { "arrow-hidden": !@arrow })
77
110
  end
78
111
 
112
+ sig { returns(HakumiComponents::Types::DataAttributes) }
79
113
  def data_attributes
80
- data = {
114
+ data = T.let({
81
115
  controller: "hakumi--popover",
82
116
  "hakumi--popover-placement-value": @placement,
83
117
  "hakumi--popover-trigger-value": normalized_triggers.join(","),
84
118
  "hakumi--popover-disabled-value": @disabled,
85
119
  "hakumi--popover-arrow-value": @arrow,
86
120
  "hakumi--popover-auto-adjust-overflow-value": @auto_adjust_overflow
87
- }
121
+ }, HakumiComponents::Types::DataAttributes)
88
122
  data[:"hakumi--popover-open-value"] = @open unless @open.nil?
123
+ existing = @html_options[:data]
124
+ existing.each { |key, value| data[key] = value } if existing.is_a?(Hash)
89
125
  data
90
126
  end
91
127
 
128
+ sig { returns(HakumiComponents::Types::Renderable) }
92
129
  def trigger_content
93
130
  content.presence || render_value(@trigger_body || @body)
94
131
  end
95
132
 
133
+ sig { returns(HakumiComponents::Types::Renderable) }
96
134
  def render_title
97
135
  render_value(@title)
98
136
  end
99
137
 
138
+ sig { returns(HakumiComponents::Types::Renderable) }
100
139
  def render_popover_content
101
140
  render_value(@body)
102
141
  end
103
142
 
104
- def has_title?
143
+ sig { returns(T::Boolean) }
144
+ def title_present?
105
145
  @title.present?
106
146
  end
107
147
 
108
- def has_popover_content?
148
+ sig { returns(T::Boolean) }
149
+ def popover_content?
109
150
  @body.present?
110
151
  end
111
152
 
112
153
  private
113
154
 
155
+ class << self
156
+ extend T::Sig
157
+
158
+ private
159
+
160
+ sig { params(value: HakumiComponents::Types::HtmlAttributeValue).returns(T.nilable(TriggerValue)) }
161
+ def trigger_param(value)
162
+ return value if value.is_a?(String) || value.is_a?(Symbol)
163
+
164
+ entries = string_or_symbol_array_html_param(value)
165
+ return entries unless entries.nil?
166
+
167
+ nil
168
+ end
169
+ end
170
+
171
+ sig { returns(T::Array[Symbol]) }
114
172
  def normalized_triggers
115
173
  return [] if @trigger.nil?
116
174
 
117
- triggers = @trigger.is_a?(Array) ? @trigger : @trigger.to_s.split(/[\s,]+/)
118
- triggers.map(&:to_s).reject(&:empty?)
175
+ if @trigger.is_a?(Array)
176
+ @trigger.map { |t| t.to_s.to_sym }
177
+ elsif @trigger.is_a?(Symbol)
178
+ [ @trigger ]
179
+ else
180
+ @trigger.to_s.split(/[\s,]+/).map(&:to_sym).reject { |t| t == :"" }
181
+ end
119
182
  end
120
183
 
184
+ sig { void }
121
185
  def validate_props!
122
186
  validate_inclusion!(:placement, PLACEMENTS)
123
187
 
@@ -0,0 +1,354 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Progress
6
+ class AttributeRenderer
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
+ DisplaySize = T.type_alias { T.nilable(BaseComponent::SizeValue) }
13
+ StepStateValue = T.type_alias { Types::HtmlPrimitive }
14
+ StepState = T.type_alias { T::Hash[Symbol, StepStateValue] }
15
+ StyleBuilder = T.type_alias { T.proc.params(styles: T.nilable(Types::StyleValue)).returns(T.nilable(String)) }
16
+
17
+ sig do
18
+ params(
19
+ type: Symbol,
20
+ percent: ProgressValue,
21
+ success_percent: T.nilable(ProgressValue),
22
+ size: DisplaySize,
23
+ stroke_width: T.nilable(Float),
24
+ stroke_color: StrokeColor,
25
+ trail_color: T.nilable(String),
26
+ stroke_linecap: Symbol,
27
+ steps: T.nilable(Integer),
28
+ gap_degree: T.nilable(Float),
29
+ gap_position: T.nilable(Symbol),
30
+ gradient_id: String,
31
+ style_builder: StyleBuilder
32
+ ).void
33
+ end
34
+ def initialize(
35
+ type:,
36
+ percent:,
37
+ success_percent:,
38
+ size:,
39
+ stroke_width:,
40
+ stroke_color:,
41
+ trail_color:,
42
+ stroke_linecap:,
43
+ steps:,
44
+ gap_degree:,
45
+ gap_position:,
46
+ gradient_id:,
47
+ style_builder:
48
+ )
49
+ @type = T.let(type, Symbol)
50
+ @percent = T.let(percent, ProgressValue)
51
+ @success_percent = T.let(success_percent, T.nilable(ProgressValue))
52
+ @size = T.let(size, DisplaySize)
53
+ @stroke_width = T.let(stroke_width, T.nilable(Float))
54
+ @stroke_color = T.let(stroke_color, StrokeColor)
55
+ @trail_color = T.let(trail_color, T.nilable(String))
56
+ @stroke_linecap = T.let(stroke_linecap, Symbol)
57
+ @steps = T.let(steps, T.nilable(Integer))
58
+ @gap_degree = T.let(gap_degree, T.nilable(Float))
59
+ @gap_position = T.let(gap_position, T.nilable(Symbol))
60
+ @gradient_id = T.let(gradient_id, String)
61
+ @style_builder = T.let(style_builder, StyleBuilder)
62
+ @circle_geometry = T.let(nil, T.nilable(Progress::CircleGeometry))
63
+ @stroke_gradient = T.let(nil, T.nilable(Progress::StrokeGradientValue))
64
+ end
65
+
66
+ sig { returns(T.nilable(String)) }
67
+ def line_style
68
+ style(
69
+ "--hakumi-progress-stroke-width" => "#{line_stroke_width}px",
70
+ "--hakumi-progress-trail-color" => @trail_color
71
+ )
72
+ end
73
+
74
+ sig { returns(T.nilable(String)) }
75
+ def line_bar_style
76
+ styles = T.let({ "width" => "#{@percent}%" }, Types::StyleMap)
77
+ if gradient?
78
+ styles["background"] = gradient_css
79
+ elsif @stroke_color.is_a?(String)
80
+ styles["background-color"] = @stroke_color
81
+ end
82
+ style(styles)
83
+ end
84
+
85
+ sig { returns(T.nilable(String)) }
86
+ def line_success_style
87
+ success = @success_percent
88
+ return nil if success.nil?
89
+
90
+ style("width" => "#{success}%")
91
+ end
92
+
93
+ sig { returns(T::Array[StepState]) }
94
+ def line_steps
95
+ steps_renderer.line_steps
96
+ end
97
+
98
+ sig { returns(Integer) }
99
+ def circle_size
100
+ return @size if @size.is_a?(Integer)
101
+
102
+ 120
103
+ end
104
+
105
+ sig { returns(T.nilable(String)) }
106
+ def circle_inner_style
107
+ style(
108
+ "width" => "#{circle_size}px",
109
+ "height" => "#{circle_size}px"
110
+ )
111
+ end
112
+
113
+ sig { returns(Types::HtmlAttributes) }
114
+ def circle_trail_attributes
115
+ return {} if circle_steps?
116
+
117
+ {
118
+ class: "hakumi-progress-circle-trail",
119
+ cx: 50,
120
+ cy: 50,
121
+ r: circle_radius,
122
+ fill: "none",
123
+ stroke: "var(--hakumi-progress-trail-color, var(--color-border-secondary))",
124
+ "stroke-width": circle_stroke_width,
125
+ "stroke-dasharray": circle_trail_dasharray,
126
+ "stroke-dashoffset": circle_dashoffset,
127
+ transform: circle_transform
128
+ }
129
+ end
130
+
131
+ sig { returns(Types::HtmlAttributes) }
132
+ def circle_path_attributes
133
+ return {} if circle_steps?
134
+
135
+ attrs = T.let(
136
+ {
137
+ class: "hakumi-progress-circle-path",
138
+ cx: 50,
139
+ cy: 50,
140
+ r: circle_radius,
141
+ fill: "none",
142
+ "stroke-linecap": @stroke_linecap,
143
+ "stroke-width": circle_stroke_width,
144
+ "stroke-dasharray": circle_path_dasharray,
145
+ "stroke-dashoffset": circle_dashoffset,
146
+ transform: circle_transform
147
+ },
148
+ Types::HtmlAttributes
149
+ )
150
+
151
+ stroke = circle_stroke
152
+ attrs[:style] = style("stroke" => stroke) if stroke
153
+ attrs
154
+ end
155
+
156
+ sig { returns(T.nilable(String)) }
157
+ def circle_trail_style
158
+ style("--hakumi-progress-trail-color" => @trail_color)
159
+ end
160
+
161
+ sig { returns(T::Array[[ String, String ]]) }
162
+ def circle_gradient_attributes
163
+ gradient = gradient_value
164
+ return [] if gradient.nil?
165
+
166
+ gradient.stop_pairs
167
+ end
168
+
169
+ sig { returns(String) }
170
+ attr_reader :gradient_id
171
+
172
+ sig { returns(Types::HtmlAttributes) }
173
+ def gradient_svg_direction
174
+ gradient = gradient_value
175
+ return Progress::StrokeGradientValue.default_svg_direction if gradient.nil?
176
+
177
+ gradient.svg_direction
178
+ end
179
+
180
+ sig { returns(T::Boolean) }
181
+ def gradient?
182
+ @stroke_color.is_a?(Hash)
183
+ end
184
+
185
+ sig { returns(T.nilable(String)) }
186
+ def gradient_css
187
+ gradient = gradient_value
188
+ return nil if gradient.nil?
189
+
190
+ gradient.css
191
+ end
192
+
193
+ sig { returns(Float) }
194
+ def circle_stroke_width
195
+ stroke_width_value.to_f
196
+ end
197
+
198
+ sig { returns(Float) }
199
+ def circle_radius
200
+ circle_geometry.radius
201
+ end
202
+
203
+ sig { returns(String) }
204
+ def circle_transform
205
+ circle_geometry.transform
206
+ end
207
+
208
+ sig { returns(T::Boolean) }
209
+ def circle_steps?
210
+ steps_renderer.circle_steps?
211
+ end
212
+
213
+ sig { returns(T::Array[StepState]) }
214
+ def circle_steps
215
+ steps_renderer(
216
+ circle_geometry: circle_geometry,
217
+ gradient_id: gradient? ? gradient_id : nil
218
+ ).circle_steps
219
+ end
220
+
221
+ sig { params(step: StepState).returns(T.nilable(String)) }
222
+ def circle_step_style(step)
223
+ stroke = step[:stroke]
224
+ style("stroke" => stroke&.to_s)
225
+ end
226
+
227
+ sig { returns(Symbol) }
228
+ def circle_gap_position
229
+ return @gap_position unless @gap_position.nil?
230
+
231
+ dashboard? ? :bottom : :top
232
+ end
233
+
234
+ sig { returns(ProgressValue) }
235
+ def stroke_width_value
236
+ return whole_number?(@stroke_width) ? @stroke_width.to_i : @stroke_width unless @stroke_width.nil?
237
+
238
+ return 6 if circle? || dashboard?
239
+
240
+ width = line_stroke_width
241
+ whole_number?(width) ? width.to_i : width
242
+ end
243
+
244
+ sig { returns(Progress::CircleGeometry) }
245
+ def circle_geometry
246
+ geometry = @circle_geometry
247
+ return geometry unless geometry.nil?
248
+
249
+ geometry = Progress::CircleGeometry.new(
250
+ stroke_width: circle_stroke_width,
251
+ gap_degree: circle_gap_degree,
252
+ gap_position: circle_gap_position
253
+ )
254
+ @circle_geometry = geometry
255
+ end
256
+
257
+ private
258
+
259
+ sig { params(styles: T.nilable(Types::StyleValue)).returns(T.nilable(String)) }
260
+ def style(styles)
261
+ @style_builder.call(styles)
262
+ end
263
+
264
+ sig { returns(T::Boolean) }
265
+ def circle?
266
+ @type == :circle
267
+ end
268
+
269
+ sig { returns(T::Boolean) }
270
+ def dashboard?
271
+ @type == :dashboard
272
+ end
273
+
274
+ sig { returns(Float) }
275
+ def line_stroke_width
276
+ return @stroke_width unless @stroke_width.nil?
277
+ return 6.0 if @size == :small
278
+
279
+ 8.0
280
+ end
281
+
282
+ sig { returns(T.nilable(String)) }
283
+ def circle_stroke
284
+ return "url(##{gradient_id})" if gradient?
285
+ return @stroke_color if @stroke_color.is_a?(String) && @stroke_color.present?
286
+
287
+ nil
288
+ end
289
+
290
+ sig { returns(Float) }
291
+ def circle_gap_degree
292
+ return @gap_degree unless @gap_degree.nil?
293
+
294
+ dashboard? ? 75.0 : 0.0
295
+ end
296
+
297
+ sig { returns(String) }
298
+ def circle_dashoffset
299
+ format_float(circle_geometry.dashoffset)
300
+ end
301
+
302
+ sig { returns(String) }
303
+ def circle_path_dasharray
304
+ circle_geometry.dasharray(circle_geometry.path_length(@percent))
305
+ end
306
+
307
+ sig { returns(String) }
308
+ def circle_trail_dasharray
309
+ circle_geometry.dasharray(circle_geometry.effective_length)
310
+ end
311
+
312
+ sig { params(value: Float).returns(String) }
313
+ def format_float(value)
314
+ format("%.2f", value)
315
+ end
316
+
317
+ sig { returns(T.nilable(Progress::StrokeGradientValue)) }
318
+ def gradient_value
319
+ gradient = @stroke_gradient
320
+ return gradient unless gradient.nil?
321
+
322
+ value = @stroke_color
323
+ return nil unless value.is_a?(Hash)
324
+
325
+ gradient = Progress::StrokeGradientValue.new(value)
326
+ @stroke_gradient = gradient
327
+ end
328
+
329
+ sig do
330
+ params(
331
+ circle_geometry: T.nilable(Progress::CircleGeometry),
332
+ gradient_id: T.nilable(String)
333
+ ).returns(Progress::StepsRenderer)
334
+ end
335
+ def steps_renderer(circle_geometry: nil, gradient_id: nil)
336
+ Progress::StepsRenderer.new(
337
+ type: @type,
338
+ percent: @percent,
339
+ steps: @steps,
340
+ stroke_color: @stroke_color,
341
+ circle_geometry: circle_geometry,
342
+ gradient_id: gradient_id
343
+ )
344
+ end
345
+
346
+ sig { params(value: T.nilable(Float)).returns(T::Boolean) }
347
+ def whole_number?(value)
348
+ return false if value.nil?
349
+
350
+ (value.to_f % 1.0).zero?
351
+ end
352
+ end
353
+ end
354
+ end
@@ -0,0 +1,76 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Progress
6
+ class CircleGeometry
7
+ extend T::Sig
8
+
9
+ ProgressValue = T.type_alias { Numeric }
10
+
11
+ sig { params(stroke_width: Float, gap_degree: Float, gap_position: Symbol).void }
12
+ def initialize(stroke_width:, gap_degree:, gap_position:)
13
+ @stroke_width = T.let(stroke_width, Float)
14
+ @gap_degree = T.let(gap_degree, Float)
15
+ @gap_position = T.let(gap_position, Symbol)
16
+ end
17
+
18
+ sig { returns(Float) }
19
+ def radius
20
+ 50 - (@stroke_width / 2.0)
21
+ end
22
+
23
+ sig { returns(Float) }
24
+ def total_length
25
+ 2 * Math::PI * radius
26
+ end
27
+
28
+ sig { returns(Float) }
29
+ def gap_length
30
+ total_length * (@gap_degree / 360.0)
31
+ end
32
+
33
+ sig { returns(Float) }
34
+ def effective_length
35
+ total_length - gap_length
36
+ end
37
+
38
+ sig { params(percent: ProgressValue).returns(Float) }
39
+ def path_length(percent)
40
+ effective_length * (percent.to_f / 100.0)
41
+ end
42
+
43
+ sig { returns(Float) }
44
+ def dashoffset
45
+ gap_length / 2.0
46
+ end
47
+
48
+ sig { params(length: Float).returns(String) }
49
+ def dasharray(length)
50
+ "#{format_float(length)} #{format_float(total_length)}"
51
+ end
52
+
53
+ sig { returns(String) }
54
+ def transform
55
+ "rotate(#{rotation} 50 50)"
56
+ end
57
+
58
+ sig { returns(Integer) }
59
+ def rotation
60
+ case @gap_position
61
+ when :left then 180
62
+ when :right then 0
63
+ when :bottom then 90
64
+ else -90
65
+ end
66
+ end
67
+
68
+ private
69
+
70
+ sig { params(value: Float).returns(String) }
71
+ def format_float(value)
72
+ format("%.2f", value)
73
+ end
74
+ end
75
+ end
76
+ end