hakumi_components 0.1.17.pre → 0.1.19.pre

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (647) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +33 -3
  3. data/README.md +218 -369
  4. data/app/assets/javascripts/hakumi_components.js +40 -34
  5. data/app/assets/stylesheets/hakumi_components.css +1 -1
  6. data/app/components/{hakumi → hakumi_components}/admin_panel/component.html.erb +2 -2
  7. data/app/components/hakumi_components/admin_panel/component.rb +50 -0
  8. data/app/components/hakumi_components/affix/component.rb +112 -0
  9. data/app/components/{hakumi → hakumi_components}/alert/component.html.erb +3 -3
  10. data/app/components/hakumi_components/alert/component.rb +134 -0
  11. data/app/components/hakumi_components/anchor/component.rb +88 -0
  12. data/app/components/{hakumi → hakumi_components}/anchor/link/component.html.erb +1 -1
  13. data/app/components/hakumi_components/anchor/link/component.rb +53 -0
  14. data/app/components/hakumi_components/autocomplete/component.html.erb +94 -0
  15. data/app/components/hakumi_components/autocomplete/component.rb +228 -0
  16. data/app/components/hakumi_components/avatar/component.rb +124 -0
  17. data/app/components/hakumi_components/badge/component.rb +139 -0
  18. data/app/components/hakumi_components/base_component.rb +391 -0
  19. data/app/components/hakumi_components/breadcrumb/component.rb +38 -0
  20. data/app/components/hakumi_components/breadcrumb/item/component.html.erb +17 -0
  21. data/app/components/hakumi_components/breadcrumb/item/component.rb +83 -0
  22. data/app/components/hakumi_components/breadcrumb/item/menu_entry.rb +71 -0
  23. data/app/components/hakumi_components/button/component.rb +163 -0
  24. data/app/components/{hakumi → hakumi_components}/calendar/component.html.erb +24 -24
  25. data/app/components/hakumi_components/calendar/component.rb +471 -0
  26. data/app/components/hakumi_components/calendar/day_cell.rb +62 -0
  27. data/app/components/hakumi_components/calendar/display_model.rb +237 -0
  28. data/app/components/hakumi_components/calendar/event_entry.rb +54 -0
  29. data/app/components/hakumi_components/calendar/locale_pack.rb +28 -0
  30. data/app/components/hakumi_components/calendar/panel_entry.rb +48 -0
  31. data/app/components/hakumi_components/calendar/week_row.rb +28 -0
  32. data/app/components/{hakumi → hakumi_components}/card/component.html.erb +2 -2
  33. data/app/components/hakumi_components/card/component.rb +121 -0
  34. data/app/components/hakumi_components/card/grid/component.html.erb +3 -0
  35. data/app/components/hakumi_components/card/grid/component.rb +43 -0
  36. data/app/components/hakumi_components/card/meta/component.html.erb +23 -0
  37. data/app/components/hakumi_components/card/meta/component.rb +64 -0
  38. data/app/components/hakumi_components/carousel/autoplay_config.rb +27 -0
  39. data/app/components/{hakumi → hakumi_components}/carousel/component.html.erb +2 -2
  40. data/app/components/hakumi_components/carousel/component.rb +165 -0
  41. data/app/components/hakumi_components/carousel/dots_config.rb +36 -0
  42. data/app/components/hakumi_components/cascader/component.html.erb +59 -0
  43. data/app/components/hakumi_components/cascader/component.rb +168 -0
  44. data/app/components/{hakumi → hakumi_components}/checkbox/component.html.erb +5 -5
  45. data/app/components/hakumi_components/checkbox/component.rb +98 -0
  46. data/app/components/hakumi_components/checkbox/group/component.html.erb +12 -0
  47. data/app/components/hakumi_components/checkbox/group/component.rb +91 -0
  48. data/app/components/hakumi_components/checkbox/group/option.rb +34 -0
  49. data/app/components/{hakumi → hakumi_components}/collapse/component.html.erb +1 -1
  50. data/app/components/hakumi_components/collapse/component.rb +133 -0
  51. data/app/components/hakumi_components/collapse/panel/component.rb +108 -0
  52. data/app/components/{hakumi → hakumi_components}/color_picker/component.html.erb +21 -22
  53. data/app/components/hakumi_components/color_picker/component.rb +279 -0
  54. data/app/components/hakumi_components/color_picker/preset_group.rb +27 -0
  55. data/app/components/hakumi_components/concerns/form_field.rb +236 -0
  56. data/app/components/hakumi_components/concerns/form_field_contract.rb +57 -0
  57. data/app/components/hakumi_components/concerns/form_field_interface.rb +16 -0
  58. data/app/components/hakumi_components/concerns/input_control.rb +111 -0
  59. data/app/components/hakumi_components/concerns/input_control_contract.rb +47 -0
  60. data/app/components/hakumi_components/concerns/input_control_interface.rb +34 -0
  61. data/app/components/hakumi_components/concerns/selection_control.rb +122 -0
  62. data/app/components/hakumi_components/concerns/selection_control_contract.rb +47 -0
  63. data/app/components/hakumi_components/concerns/selection_control_interface.rb +34 -0
  64. data/app/components/{hakumi → hakumi_components}/container/component.rb +31 -16
  65. data/app/components/{hakumi → hakumi_components}/date_picker/component.html.erb +4 -6
  66. data/app/components/hakumi_components/date_picker/component.rb +254 -0
  67. data/app/components/{hakumi → hakumi_components}/date_picker/range_picker.html.erb +4 -4
  68. data/app/components/hakumi_components/date_picker/range_picker.rb +267 -0
  69. data/app/components/hakumi_components/date_picker/shared_rendering.rb +289 -0
  70. data/app/components/{hakumi → hakumi_components}/descriptions/component.html.erb +6 -6
  71. data/app/components/hakumi_components/descriptions/component.rb +219 -0
  72. data/app/components/hakumi_components/descriptions/item/component.rb +52 -0
  73. data/app/components/hakumi_components/divider/component.rb +86 -0
  74. data/app/components/{hakumi → hakumi_components}/drawer/component.html.erb +3 -3
  75. data/app/components/hakumi_components/drawer/component.rb +191 -0
  76. data/app/components/hakumi_components/dropdown/component.rb +94 -0
  77. data/app/components/{hakumi → hakumi_components}/dropdown/divider/component.rb +6 -2
  78. data/app/components/hakumi_components/dropdown/item/component.rb +73 -0
  79. data/app/components/{hakumi → hakumi_components}/empty/component.rb +68 -33
  80. data/app/components/hakumi_components/flex/component.rb +96 -0
  81. data/app/components/hakumi_components/float_button/back_top/component.rb +86 -0
  82. data/app/components/hakumi_components/float_button/component.rb +289 -0
  83. data/app/components/hakumi_components/float_button/group/component.rb +214 -0
  84. data/app/components/hakumi_components/float_button/group_cluster/component.rb +133 -0
  85. data/app/components/hakumi_components/float_button/group_spec.rb +259 -0
  86. data/app/components/hakumi_components/float_button/item_spec.rb +115 -0
  87. data/app/components/hakumi_components/float_button/position_config.rb +68 -0
  88. data/app/components/{hakumi → hakumi_components}/form/item/component.rb +36 -29
  89. data/app/components/hakumi_components/grid/col/component.rb +169 -0
  90. data/app/components/hakumi_components/grid/row/component.rb +97 -0
  91. data/app/components/{hakumi → hakumi_components}/icon/component.rb +53 -53
  92. data/app/components/{hakumi → hakumi_components}/image/component.html.erb +17 -17
  93. data/app/components/hakumi_components/image/component.rb +175 -0
  94. data/app/components/{hakumi → hakumi_components}/image/preview_group/component.html.erb +10 -10
  95. data/app/components/hakumi_components/image/preview_group/component.rb +75 -0
  96. data/app/components/hakumi_components/image/preview_group/item.rb +48 -0
  97. data/app/components/{hakumi → hakumi_components}/input/component.html.erb +1 -1
  98. data/app/components/hakumi_components/input/component.rb +261 -0
  99. data/app/components/hakumi_components/input/password/component.rb +84 -0
  100. data/app/components/{hakumi → hakumi_components}/input/text_area/component.html.erb +2 -2
  101. data/app/components/hakumi_components/input/text_area/component.rb +178 -0
  102. data/app/components/{hakumi → hakumi_components}/input_number/component.html.erb +9 -9
  103. data/app/components/hakumi_components/input_number/component.rb +280 -0
  104. data/app/components/hakumi_components/layout/component.html.erb +1 -0
  105. data/app/components/hakumi_components/layout/component.rb +28 -0
  106. data/app/components/hakumi_components/layout/content/component.html.erb +1 -0
  107. data/app/components/hakumi_components/layout/content/component.rb +29 -0
  108. data/app/components/hakumi_components/layout/footer/component.html.erb +1 -0
  109. data/app/components/hakumi_components/layout/footer/component.rb +29 -0
  110. data/app/components/hakumi_components/layout/header/component.html.erb +1 -0
  111. data/app/components/hakumi_components/layout/header/component.rb +29 -0
  112. data/app/components/hakumi_components/layout/sider/component.html.erb +1 -0
  113. data/app/components/hakumi_components/layout/sider/component.rb +90 -0
  114. data/app/components/hakumi_components/mentions/coercion.rb +46 -0
  115. data/app/components/{hakumi → hakumi_components}/mentions/component.html.erb +6 -6
  116. data/app/components/hakumi_components/mentions/component.rb +229 -0
  117. data/app/components/hakumi_components/mentions/option.rb +30 -0
  118. data/app/components/hakumi_components/menu/component.rb +117 -0
  119. data/app/components/{hakumi → hakumi_components}/menu/divider/component.rb +6 -2
  120. data/app/components/hakumi_components/menu/group/component.rb +48 -0
  121. data/app/components/hakumi_components/menu/item/component.rb +70 -0
  122. data/app/components/hakumi_components/menu/sub_menu/component.rb +104 -0
  123. data/app/components/hakumi_components/message/component.rb +141 -0
  124. data/app/components/{hakumi → hakumi_components}/modal/component.html.erb +7 -7
  125. data/app/components/hakumi_components/modal/component.rb +152 -0
  126. data/app/components/hakumi_components/modal/confirm/component.html.erb +23 -0
  127. data/app/components/hakumi_components/modal/confirm/component.rb +87 -0
  128. data/app/components/hakumi_components/modal/error/component.rb +20 -0
  129. data/app/components/hakumi_components/modal/info/component.rb +17 -0
  130. data/app/components/hakumi_components/modal/status_component.rb +150 -0
  131. data/app/components/hakumi_components/modal/success/component.rb +17 -0
  132. data/app/components/hakumi_components/modal/warning/component.rb +26 -0
  133. data/app/components/hakumi_components/notification/component.rb +160 -0
  134. data/app/components/{hakumi → hakumi_components}/pagination/component.html.erb +8 -12
  135. data/app/components/hakumi_components/pagination/component.rb +238 -0
  136. data/app/components/hakumi_components/pagination/page_item.rb +49 -0
  137. data/app/components/{hakumi → hakumi_components}/popconfirm/component.html.erb +2 -2
  138. data/app/components/hakumi_components/popconfirm/component.rb +120 -0
  139. data/app/components/{hakumi → hakumi_components}/popover/component.html.erb +2 -2
  140. data/app/components/hakumi_components/popover/component.rb +196 -0
  141. data/app/components/hakumi_components/progress/attribute_renderer.rb +354 -0
  142. data/app/components/hakumi_components/progress/circle_geometry.rb +76 -0
  143. data/app/components/{hakumi → hakumi_components}/progress/component.html.erb +19 -18
  144. data/app/components/hakumi_components/progress/component.rb +393 -0
  145. data/app/components/hakumi_components/progress/controller_locals_parser.rb +130 -0
  146. data/app/components/hakumi_components/progress/info_tooltip_policy.rb +39 -0
  147. data/app/components/hakumi_components/progress/info_value.rb +75 -0
  148. data/app/components/hakumi_components/progress/status_state.rb +31 -0
  149. data/app/components/hakumi_components/progress/steps_renderer.rb +149 -0
  150. data/app/components/hakumi_components/progress/stroke_gradient_value.rb +90 -0
  151. data/app/components/hakumi_components/qr_code/component.rb +405 -0
  152. data/app/components/{hakumi → hakumi_components}/radio/component.html.erb +3 -3
  153. data/app/components/hakumi_components/radio/component.rb +113 -0
  154. data/app/components/{hakumi → hakumi_components}/radio/group/component.html.erb +10 -10
  155. data/app/components/hakumi_components/radio/group/component.rb +181 -0
  156. data/app/components/hakumi_components/radio/group/option.rb +43 -0
  157. data/app/components/{hakumi → hakumi_components}/rate/component.html.erb +2 -2
  158. data/app/components/{hakumi → hakumi_components}/rate/component.rb +58 -12
  159. data/app/components/hakumi_components/result/component.rb +175 -0
  160. data/app/components/{hakumi → hakumi_components}/segmented/component.html.erb +1 -1
  161. data/app/components/hakumi_components/segmented/component.rb +232 -0
  162. data/app/components/hakumi_components/segmented/option.rb +191 -0
  163. data/app/components/{hakumi → hakumi_components}/select/component.html.erb +22 -22
  164. data/app/components/hakumi_components/select/component.rb +176 -0
  165. data/app/components/hakumi_components/selection_control/coercion.rb +161 -0
  166. data/app/components/hakumi_components/selection_control/entry.rb +22 -0
  167. data/app/components/hakumi_components/selection_control/option.rb +42 -0
  168. data/app/components/hakumi_components/selection_control/option_group.rb +34 -0
  169. data/app/components/hakumi_components/selection_control/tree_node.rb +123 -0
  170. data/app/components/hakumi_components/skeleton/avatar/component.rb +88 -0
  171. data/app/components/hakumi_components/skeleton/avatar_config.rb +79 -0
  172. data/app/components/hakumi_components/skeleton/button/component.rb +77 -0
  173. data/app/components/hakumi_components/skeleton/component.rb +186 -0
  174. data/app/components/{hakumi → hakumi_components}/skeleton/image/component.html.erb +1 -1
  175. data/app/components/hakumi_components/skeleton/image/component.rb +104 -0
  176. data/app/components/hakumi_components/skeleton/input/component.rb +66 -0
  177. data/app/components/hakumi_components/skeleton/node/component.rb +59 -0
  178. data/app/components/hakumi_components/skeleton/paragraph_config.rb +92 -0
  179. data/app/components/hakumi_components/skeleton/title_config.rb +31 -0
  180. data/app/components/{hakumi → hakumi_components}/slider/component.html.erb +11 -11
  181. data/app/components/hakumi_components/slider/component.rb +347 -0
  182. data/app/components/hakumi_components/slider/mark.rb +27 -0
  183. data/app/components/hakumi_components/space/compact/component.html.erb +1 -0
  184. data/app/components/hakumi_components/space/compact/component.rb +52 -0
  185. data/app/components/{hakumi → hakumi_components}/space/component.html.erb +1 -1
  186. data/app/components/hakumi_components/space/component.rb +109 -0
  187. data/app/components/hakumi_components/spin/component.rb +270 -0
  188. data/app/components/hakumi_components/splitter/component.html.erb +9 -0
  189. data/app/components/hakumi_components/splitter/component.rb +71 -0
  190. data/app/components/hakumi_components/splitter/panel/component.html.erb +1 -0
  191. data/app/components/hakumi_components/splitter/panel/component.rb +98 -0
  192. data/app/components/hakumi_components/statistic/component.rb +371 -0
  193. data/app/components/{hakumi → hakumi_components}/steps/component.html.erb +4 -4
  194. data/app/components/hakumi_components/steps/component.rb +177 -0
  195. data/app/components/hakumi_components/steps/item/component.rb +130 -0
  196. data/app/components/{hakumi → hakumi_components}/switch/component.html.erb +2 -3
  197. data/app/components/hakumi_components/switch/component.rb +150 -0
  198. data/app/components/hakumi_components/table/column/component.rb +137 -0
  199. data/app/components/hakumi_components/table/column_definition.rb +181 -0
  200. data/app/components/{hakumi → hakumi_components}/table/column_group/component.rb +35 -24
  201. data/app/components/{hakumi → hakumi_components}/table/component.html.erb +22 -31
  202. data/app/components/hakumi_components/table/component.rb +470 -0
  203. data/app/components/hakumi_components/table/concerns/attribute_helpers.rb +22 -0
  204. data/app/components/hakumi_components/table/concerns/cell_rendering.rb +277 -0
  205. data/app/components/hakumi_components/table/concerns/columns.rb +421 -0
  206. data/app/components/hakumi_components/table/concerns/editable.rb +68 -0
  207. data/app/components/hakumi_components/table/concerns/ellipsis.rb +64 -0
  208. data/app/components/hakumi_components/table/concerns/fixed_columns.rb +94 -0
  209. data/app/components/hakumi_components/table/concerns/surface_rendering.rb +365 -0
  210. data/app/components/hakumi_components/table/configs.rb +518 -0
  211. data/app/components/hakumi_components/table/definition_types.rb +40 -0
  212. data/app/components/hakumi_components/table/display_rows.rb +56 -0
  213. data/app/components/hakumi_components/table/ellipsis_config.rb +143 -0
  214. data/app/components/hakumi_components/table/fixed_offset.rb +36 -0
  215. data/app/components/hakumi_components/table/header_cell.rb +23 -0
  216. data/app/components/hakumi_components/table/row_record.rb +60 -0
  217. data/app/components/hakumi_components/table/row_render_state.rb +85 -0
  218. data/app/components/{hakumi → hakumi_components}/tabs/component.html.erb +11 -11
  219. data/app/components/hakumi_components/tabs/component.rb +246 -0
  220. data/app/components/hakumi_components/tabs/indicator_config.rb +34 -0
  221. data/app/components/hakumi_components/tabs/item/component.rb +147 -0
  222. data/app/components/hakumi_components/tag/component.rb +282 -0
  223. data/app/components/{hakumi → hakumi_components}/tag/group/component.rb +20 -8
  224. data/app/components/{hakumi → hakumi_components}/time_picker/component.html.erb +10 -10
  225. data/app/components/{hakumi → hakumi_components}/time_picker/component.rb +75 -21
  226. data/app/components/hakumi_components/timeline/component.html.erb +8 -0
  227. data/app/components/hakumi_components/timeline/component.rb +301 -0
  228. data/app/components/{hakumi → hakumi_components}/timeline/item/component.html.erb +2 -2
  229. data/app/components/hakumi_components/timeline/item/component.rb +198 -0
  230. data/app/components/{hakumi → hakumi_components}/tooltip/component.html.erb +1 -1
  231. data/app/components/hakumi_components/tooltip/component.rb +146 -0
  232. data/app/components/hakumi_components/tour/component.rb +183 -0
  233. data/app/components/hakumi_components/tour/step.rb +86 -0
  234. data/app/components/hakumi_components/transfer/coercion.rb +66 -0
  235. data/app/components/hakumi_components/transfer/component.html.erb +204 -0
  236. data/app/components/hakumi_components/transfer/component.rb +395 -0
  237. data/app/components/hakumi_components/transfer/item.rb +54 -0
  238. data/app/components/hakumi_components/transfer/operations.rb +31 -0
  239. data/app/components/{hakumi → hakumi_components}/tree/component.html.erb +1 -1
  240. data/app/components/hakumi_components/tree/component.rb +342 -0
  241. data/app/components/hakumi_components/tree/concerns/node_rendering.rb +365 -0
  242. data/app/components/hakumi_components/tree/node.rb +276 -0
  243. data/app/components/{hakumi → hakumi_components}/tree_select/component.html.erb +27 -13
  244. data/app/components/hakumi_components/tree_select/component.rb +270 -0
  245. data/app/components/hakumi_components/typography/base_component.rb +142 -0
  246. data/app/components/hakumi_components/typography/link/component.rb +91 -0
  247. data/app/components/{hakumi → hakumi_components}/typography/paragraph/component.rb +7 -1
  248. data/{sig/hakumi/typography/text/component.rbs → app/components/hakumi_components/typography/text/component.rb} +3 -1
  249. data/app/components/hakumi_components/typography/title/component.rb +84 -0
  250. data/app/components/hakumi_components/upload/component.html.erb +135 -0
  251. data/app/components/hakumi_components/upload/component.rb +367 -0
  252. data/app/components/hakumi_components/upload/file_entry.rb +53 -0
  253. data/app/controllers/{hakumi → hakumi_components}/components_controller.rb +3 -2
  254. data/app/form_builders/hakumi_components/form_builder/contracts.rb +55 -0
  255. data/app/form_builders/hakumi_components/form_builder/field_context.rb +43 -0
  256. data/app/form_builders/hakumi_components/form_builder/model_binding.rb +163 -0
  257. data/app/form_builders/hakumi_components/form_builder.rb +403 -0
  258. data/app/helpers/hakumi_components/form_helper.rb +43 -0
  259. data/app/javascript/hakumi_components/controllers/hakumi/admin_panel_controller.js +5 -7
  260. data/app/javascript/hakumi_components/controllers/hakumi/back_top_controller.js +1 -1
  261. data/app/javascript/hakumi_components/controllers/hakumi/button_controller.js +108 -2
  262. data/app/javascript/hakumi_components/controllers/hakumi/calendar_controller.js +183 -95
  263. data/app/javascript/hakumi_components/controllers/hakumi/color_picker_controller.js +23 -285
  264. data/app/javascript/hakumi_components/controllers/hakumi/date_picker_controller.js +274 -262
  265. data/app/javascript/hakumi_components/controllers/hakumi/float_button_group_controller.js +2 -2
  266. data/app/javascript/hakumi_components/controllers/hakumi/message_controller.js +4 -2
  267. data/app/javascript/hakumi_components/controllers/hakumi/modal_controller.js +119 -125
  268. data/app/javascript/hakumi_components/controllers/hakumi/table/editable.js +291 -0
  269. data/app/javascript/hakumi_components/controllers/hakumi/table_controller.js +166 -366
  270. data/app/javascript/hakumi_components/controllers/hakumi/tabs_controller.js +8 -2
  271. data/app/javascript/hakumi_components/controllers/hakumi/tag_controller.js +27 -25
  272. data/app/javascript/hakumi_components/controllers/hakumi/tag_group_controller.js +19 -18
  273. data/app/javascript/hakumi_components/controllers/hakumi/theme_controller.js +116 -117
  274. data/app/javascript/hakumi_components/controllers/hakumi/transfer_controller.js +17 -1
  275. data/app/javascript/hakumi_components/controllers/hakumi/tree_controller.js +363 -78
  276. data/app/javascript/hakumi_components/controllers/hakumi/typography_controller.js +3 -3
  277. data/app/javascript/hakumi_components/controllers/hakumi/upload_controller.js +320 -204
  278. data/app/javascript/hakumi_components/core/render_component.js +37 -11
  279. data/app/javascript/hakumi_components/utils/color_helper.js +262 -0
  280. data/app/javascript/stylesheets/_base.scss +9 -0
  281. data/app/javascript/stylesheets/_hakumi_components.scss +1 -0
  282. data/app/javascript/stylesheets/components/_breadcrumb.scss +2 -2
  283. data/app/javascript/stylesheets/components/_calendar.scss +13 -13
  284. data/app/javascript/stylesheets/components/_cascader.scss +5 -5
  285. data/app/javascript/stylesheets/components/_checkbox.scss +9 -11
  286. data/app/javascript/stylesheets/components/_color_picker.scss +11 -11
  287. data/app/javascript/stylesheets/components/_date_picker.scss +4 -4
  288. data/app/javascript/stylesheets/components/_descriptions.scss +2 -2
  289. data/app/javascript/stylesheets/components/_drawer.scss +3 -3
  290. data/app/javascript/stylesheets/components/_dropdown.scss +2 -2
  291. data/app/javascript/stylesheets/components/_flex.scss +1 -1
  292. data/app/javascript/stylesheets/components/_float_button.scss +5 -5
  293. data/app/javascript/stylesheets/components/_form_item.scss +92 -0
  294. data/app/javascript/stylesheets/components/_image.scss +15 -15
  295. data/app/javascript/stylesheets/components/_input.scss +23 -113
  296. data/app/javascript/stylesheets/components/_layout.scss +27 -26
  297. data/app/javascript/stylesheets/components/_menu.scss +15 -15
  298. data/app/javascript/stylesheets/components/_modal.scss +13 -13
  299. data/app/javascript/stylesheets/components/_notification.scss +3 -3
  300. data/app/javascript/stylesheets/components/_popover.scss +1 -1
  301. data/app/javascript/stylesheets/components/_segmented.scss +3 -3
  302. data/app/javascript/stylesheets/components/_select.scss +6 -6
  303. data/app/javascript/stylesheets/components/_slider.scss +1 -1
  304. data/app/javascript/stylesheets/components/_spin.scss +2 -2
  305. data/app/javascript/stylesheets/components/_steps.scss +10 -10
  306. data/app/javascript/stylesheets/components/_switch.scss +11 -10
  307. data/app/javascript/stylesheets/components/_table.scss +6 -6
  308. data/app/javascript/stylesheets/components/_tag.scss +2 -2
  309. data/app/javascript/stylesheets/components/_tooltip.scss +4 -4
  310. data/app/javascript/stylesheets/components/_tree_select.scss +3 -3
  311. data/app/javascript/stylesheets/components/_typography.scss +3 -3
  312. data/app/javascript/stylesheets/components/_upload.scss +4 -4
  313. data/app/services/hakumi_components/component_handler.rb +124 -0
  314. data/app/services/hakumi_components/icon/loader.rb +112 -0
  315. data/app/services/hakumi_components/illustrations/loader.rb +51 -0
  316. data/app/views/hakumi/_admin_panel.html.erb +1 -1
  317. data/app/views/hakumi/_affix.html.erb +1 -1
  318. data/app/views/hakumi/_alert.html.erb +4 -1
  319. data/app/views/hakumi/_confirm.html.erb +1 -1
  320. data/app/views/hakumi/_drawer.html.erb +2 -2
  321. data/app/views/hakumi/_message.html.erb +1 -1
  322. data/app/views/hakumi/_modal.html.erb +1 -1
  323. data/app/views/hakumi/_notification.html.erb +1 -1
  324. data/app/views/hakumi/_popconfirm.html.erb +1 -1
  325. data/app/views/hakumi/_popover.html.erb +1 -1
  326. data/app/views/hakumi/_qr_code.html.erb +1 -1
  327. data/app/views/hakumi/_result.html.erb +1 -1
  328. data/app/views/hakumi/_segmented.html.erb +1 -1
  329. data/app/views/hakumi/_skeleton.html.erb +1 -1
  330. data/app/views/hakumi/_spin.html.erb +1 -1
  331. data/app/views/hakumi/_statistic.html.erb +12 -12
  332. data/app/views/hakumi/_table.html.erb +1 -1
  333. data/app/views/hakumi/_tag.html.erb +1 -1
  334. data/app/views/hakumi/_timeline.html.erb +1 -1
  335. data/app/views/hakumi/_tree.html.erb +1 -1
  336. data/lib/generators/{hakumi → hakumi_components}/install_generator.rb +17 -4
  337. data/lib/hakumi_components/documentation/models.rb +290 -0
  338. data/lib/hakumi_components/documentation/node.rb +73 -0
  339. data/lib/hakumi_components/documentation.rb +141 -64
  340. data/lib/hakumi_components/engine.rb +26 -14
  341. data/lib/hakumi_components/file_size_checker.rb +215 -0
  342. data/lib/hakumi_components/locales/en.yml +115 -0
  343. data/lib/hakumi_components/rails/attribute_introspection.rb +40 -108
  344. data/lib/hakumi_components/rails/model_reflection.rb +154 -0
  345. data/lib/hakumi_components/rails/validation_introspection.rb +26 -74
  346. data/lib/hakumi_components/rails/validation_mapper.rb +149 -202
  347. data/lib/hakumi_components/rails.rb +3 -1
  348. data/lib/hakumi_components/stylesheet_ownership_checker.rb +191 -0
  349. data/lib/hakumi_components/types/form_field.rb +24 -0
  350. data/lib/hakumi_components/types/html.rb +28 -0
  351. data/lib/hakumi_components/types/rendering.rb +10 -0
  352. data/lib/hakumi_components/types/stimulus.rb +19 -0
  353. data/lib/hakumi_components/types/validation.rb +22 -0
  354. data/lib/hakumi_components/types.rb +13 -0
  355. data/lib/hakumi_components/version.rb +3 -2
  356. data/lib/hakumi_components.rb +26 -5
  357. data/lib/tasks/coverage.rake +1 -0
  358. metadata +310 -346
  359. data/app/components/hakumi/admin_panel/component.rb +0 -40
  360. data/app/components/hakumi/affix/component.rb +0 -88
  361. data/app/components/hakumi/alert/component.rb +0 -84
  362. data/app/components/hakumi/anchor/component.rb +0 -80
  363. data/app/components/hakumi/anchor/link/component.rb +0 -34
  364. data/app/components/hakumi/autocomplete/component.html.erb +0 -74
  365. data/app/components/hakumi/autocomplete/component.rb +0 -202
  366. data/app/components/hakumi/avatar/component.rb +0 -91
  367. data/app/components/hakumi/badge/component.rb +0 -85
  368. data/app/components/hakumi/base_component.rb +0 -192
  369. data/app/components/hakumi/breadcrumb/component.rb +0 -27
  370. data/app/components/hakumi/breadcrumb/item/component.html.erb +0 -17
  371. data/app/components/hakumi/breadcrumb/item/component.rb +0 -35
  372. data/app/components/hakumi/button/component.rb +0 -121
  373. data/app/components/hakumi/calendar/component.rb +0 -448
  374. data/app/components/hakumi/card/component.rb +0 -94
  375. data/app/components/hakumi/card/grid/component.html.erb +0 -3
  376. data/app/components/hakumi/card/grid/component.rb +0 -37
  377. data/app/components/hakumi/card/meta/component.html.erb +0 -23
  378. data/app/components/hakumi/card/meta/component.rb +0 -46
  379. data/app/components/hakumi/carousel/component.rb +0 -147
  380. data/app/components/hakumi/cascader/component.html.erb +0 -41
  381. data/app/components/hakumi/cascader/component.rb +0 -119
  382. data/app/components/hakumi/checkbox/component.rb +0 -64
  383. data/app/components/hakumi/checkbox/group/component.html.erb +0 -12
  384. data/app/components/hakumi/checkbox/group/component.rb +0 -64
  385. data/app/components/hakumi/collapse/component.rb +0 -117
  386. data/app/components/hakumi/collapse/panel/component.rb +0 -65
  387. data/app/components/hakumi/color_picker/component.rb +0 -169
  388. data/app/components/hakumi/concerns/form_field.rb +0 -135
  389. data/app/components/hakumi/date_picker/component.rb +0 -140
  390. data/app/components/hakumi/date_picker/range_picker.rb +0 -136
  391. data/app/components/hakumi/date_picker/shared_rendering.rb +0 -199
  392. data/app/components/hakumi/descriptions/component.rb +0 -182
  393. data/app/components/hakumi/descriptions/item/component.rb +0 -28
  394. data/app/components/hakumi/divider/component.rb +0 -66
  395. data/app/components/hakumi/drawer/component.rb +0 -108
  396. data/app/components/hakumi/dropdown/component.rb +0 -97
  397. data/app/components/hakumi/dropdown/item/component.rb +0 -47
  398. data/app/components/hakumi/flex/component.rb +0 -78
  399. data/app/components/hakumi/float_button/back_top/component.rb +0 -79
  400. data/app/components/hakumi/float_button/component.rb +0 -203
  401. data/app/components/hakumi/float_button/group/component.rb +0 -227
  402. data/app/components/hakumi/float_button/group_cluster/component.rb +0 -141
  403. data/app/components/hakumi/grid/col/component.rb +0 -98
  404. data/app/components/hakumi/grid/row/component.rb +0 -79
  405. data/app/components/hakumi/image/component.rb +0 -112
  406. data/app/components/hakumi/image/preview_group/component.rb +0 -70
  407. data/app/components/hakumi/input/component.rb +0 -185
  408. data/app/components/hakumi/input/password/component.rb +0 -95
  409. data/app/components/hakumi/input/text_area/component.rb +0 -131
  410. data/app/components/hakumi/input_number/component.rb +0 -198
  411. data/app/components/hakumi/layout/component.html.erb +0 -1
  412. data/app/components/hakumi/layout/component.rb +0 -21
  413. data/app/components/hakumi/layout/content/component.html.erb +0 -1
  414. data/app/components/hakumi/layout/content/component.rb +0 -19
  415. data/app/components/hakumi/layout/footer/component.html.erb +0 -1
  416. data/app/components/hakumi/layout/footer/component.rb +0 -19
  417. data/app/components/hakumi/layout/header/component.html.erb +0 -1
  418. data/app/components/hakumi/layout/header/component.rb +0 -19
  419. data/app/components/hakumi/layout/sider/component.html.erb +0 -1
  420. data/app/components/hakumi/layout/sider/component.rb +0 -56
  421. data/app/components/hakumi/mentions/component.rb +0 -165
  422. data/app/components/hakumi/menu/component.rb +0 -126
  423. data/app/components/hakumi/menu/group/component.rb +0 -47
  424. data/app/components/hakumi/menu/item/component.rb +0 -47
  425. data/app/components/hakumi/menu/sub_menu/component.rb +0 -92
  426. data/app/components/hakumi/message/component.rb +0 -106
  427. data/app/components/hakumi/modal/component.rb +0 -99
  428. data/app/components/hakumi/modal/confirm/component.html.erb +0 -23
  429. data/app/components/hakumi/modal/confirm/component.rb +0 -46
  430. data/app/components/hakumi/modal/error/component.rb +0 -51
  431. data/app/components/hakumi/modal/info/component.rb +0 -51
  432. data/app/components/hakumi/modal/success/component.rb +0 -51
  433. data/app/components/hakumi/modal/warning/component.rb +0 -56
  434. data/app/components/hakumi/notification/component.rb +0 -126
  435. data/app/components/hakumi/pagination/component.rb +0 -205
  436. data/app/components/hakumi/popconfirm/component.rb +0 -72
  437. data/app/components/hakumi/popover/component.rb +0 -132
  438. data/app/components/hakumi/progress/component.rb +0 -565
  439. data/app/components/hakumi/qr_code/component.rb +0 -339
  440. data/app/components/hakumi/radio/component.rb +0 -80
  441. data/app/components/hakumi/radio/group/component.rb +0 -132
  442. data/app/components/hakumi/result/component.rb +0 -125
  443. data/app/components/hakumi/segmented/component.rb +0 -269
  444. data/app/components/hakumi/select/component.rb +0 -125
  445. data/app/components/hakumi/skeleton/avatar/component.rb +0 -66
  446. data/app/components/hakumi/skeleton/button/component.rb +0 -59
  447. data/app/components/hakumi/skeleton/component.rb +0 -200
  448. data/app/components/hakumi/skeleton/image/component.rb +0 -80
  449. data/app/components/hakumi/skeleton/input/component.rb +0 -51
  450. data/app/components/hakumi/skeleton/node/component.rb +0 -43
  451. data/app/components/hakumi/slider/component.rb +0 -228
  452. data/app/components/hakumi/space/compact/component.html.erb +0 -3
  453. data/app/components/hakumi/space/compact/component.rb +0 -36
  454. data/app/components/hakumi/space/component.rb +0 -73
  455. data/app/components/hakumi/spin/component.rb +0 -209
  456. data/app/components/hakumi/splitter/component.html.erb +0 -3
  457. data/app/components/hakumi/splitter/component.rb +0 -38
  458. data/app/components/hakumi/splitter/panel/component.html.erb +0 -3
  459. data/app/components/hakumi/splitter/panel/component.rb +0 -73
  460. data/app/components/hakumi/statistic/component.rb +0 -287
  461. data/app/components/hakumi/steps/component.rb +0 -145
  462. data/app/components/hakumi/steps/item/component.rb +0 -91
  463. data/app/components/hakumi/switch/component.rb +0 -124
  464. data/app/components/hakumi/table/column/component.rb +0 -80
  465. data/app/components/hakumi/table/component.rb +0 -731
  466. data/app/components/hakumi/table/concerns/columns.rb +0 -219
  467. data/app/components/hakumi/table/concerns/editable.rb +0 -121
  468. data/app/components/hakumi/table/concerns/ellipsis.rb +0 -63
  469. data/app/components/hakumi/table/concerns/fixed_columns.rb +0 -87
  470. data/app/components/hakumi/tabs/component.rb +0 -196
  471. data/app/components/hakumi/tabs/item/component.rb +0 -87
  472. data/app/components/hakumi/tag/component.rb +0 -187
  473. data/app/components/hakumi/timeline/component.html.erb +0 -7
  474. data/app/components/hakumi/timeline/component.rb +0 -168
  475. data/app/components/hakumi/timeline/item/component.rb +0 -137
  476. data/app/components/hakumi/tooltip/component.rb +0 -121
  477. data/app/components/hakumi/tour/component.rb +0 -158
  478. data/app/components/hakumi/transfer/component.html.erb +0 -173
  479. data/app/components/hakumi/transfer/component.rb +0 -245
  480. data/app/components/hakumi/tree/component.rb +0 -523
  481. data/app/components/hakumi/tree_select/component.rb +0 -261
  482. data/app/components/hakumi/typography/base_component.rb +0 -109
  483. data/app/components/hakumi/typography/link/component.rb +0 -34
  484. data/app/components/hakumi/typography/text/component.rb +0 -8
  485. data/app/components/hakumi/typography/title/component.rb +0 -29
  486. data/app/components/hakumi/upload/component.html.erb +0 -117
  487. data/app/components/hakumi/upload/component.rb +0 -193
  488. data/app/form_builders/hakumi/form_builder.rb +0 -482
  489. data/app/helpers/hakumi/form_helper.rb +0 -39
  490. data/app/services/hakumi/component_handler.rb +0 -53
  491. data/app/services/hakumi/icon/loader.rb +0 -95
  492. data/app/services/hakumi/illustrations/loader.rb +0 -43
  493. data/sig/action_view/tag_builder.rbs +0 -24
  494. data/sig/active_support/concern_patch.rbs +0 -7
  495. data/sig/generators/hakumi/install_generator.rbs +0 -11
  496. data/sig/hakumi/admin_panel/component.rbs +0 -28
  497. data/sig/hakumi/affix/component.rbs +0 -44
  498. data/sig/hakumi/alert/component.rbs +0 -64
  499. data/sig/hakumi/anchor/component.rbs +0 -45
  500. data/sig/hakumi/anchor/link/component.rbs +0 -31
  501. data/sig/hakumi/autocomplete/component.rbs +0 -68
  502. data/sig/hakumi/avatar/component.rbs +0 -44
  503. data/sig/hakumi/badge/component.rbs +0 -41
  504. data/sig/hakumi/base_component.rbs +0 -82
  505. data/sig/hakumi/breadcrumb/component.rbs +0 -21
  506. data/sig/hakumi/breadcrumb/item/component.rbs +0 -36
  507. data/sig/hakumi/button/component.rbs +0 -56
  508. data/sig/hakumi/calendar/component.rbs +0 -41
  509. data/sig/hakumi/card/component.rbs +0 -50
  510. data/sig/hakumi/card/grid/component.rbs +0 -21
  511. data/sig/hakumi/card/meta/component.rbs +0 -25
  512. data/sig/hakumi/carousel/component.rbs +0 -88
  513. data/sig/hakumi/cascader/component.rbs +0 -52
  514. data/sig/hakumi/checkbox/component.rbs +0 -46
  515. data/sig/hakumi/checkbox/group/component.rbs +0 -34
  516. data/sig/hakumi/collapse/component.rbs +0 -57
  517. data/sig/hakumi/collapse/panel/component.rbs +0 -47
  518. data/sig/hakumi/color_picker/component.rbs +0 -80
  519. data/sig/hakumi/concerns/form_field.rbs +0 -89
  520. data/sig/hakumi/container/component.rbs +0 -37
  521. data/sig/hakumi/date_picker/component.rbs +0 -72
  522. data/sig/hakumi/date_picker/range_picker.rbs +0 -68
  523. data/sig/hakumi/date_picker/shared_rendering.rbs +0 -41
  524. data/sig/hakumi/descriptions/component.rbs +0 -59
  525. data/sig/hakumi/descriptions/item/component.rbs +0 -31
  526. data/sig/hakumi/divider/component.rbs +0 -39
  527. data/sig/hakumi/drawer/component.rbs +0 -66
  528. data/sig/hakumi/dropdown/component.rbs +0 -45
  529. data/sig/hakumi/dropdown/divider/component.rbs +0 -11
  530. data/sig/hakumi/dropdown/item/component.rbs +0 -45
  531. data/sig/hakumi/empty/component.rbs +0 -40
  532. data/sig/hakumi/flex/component.rbs +0 -47
  533. data/sig/hakumi/float_button/back_top/component.rbs +0 -45
  534. data/sig/hakumi/float_button/component.rbs +0 -80
  535. data/sig/hakumi/float_button/group/component.rbs +0 -95
  536. data/sig/hakumi/float_button/group_cluster/component.rbs +0 -60
  537. data/sig/hakumi/form/item/component.rbs +0 -39
  538. data/sig/hakumi/form_builder.rbs +0 -37
  539. data/sig/hakumi/grid/col/component.rbs +0 -50
  540. data/sig/hakumi/grid/row/component.rbs +0 -46
  541. data/sig/hakumi/icon/component.rbs +0 -56
  542. data/sig/hakumi/image/component.rbs +0 -56
  543. data/sig/hakumi/image/preview_group/component.rbs +0 -28
  544. data/sig/hakumi/input/component.rbs +0 -71
  545. data/sig/hakumi/input/password/component.rbs +0 -27
  546. data/sig/hakumi/input/text_area/component.rbs +0 -42
  547. data/sig/hakumi/input_number/component.rbs +0 -85
  548. data/sig/hakumi/layout/component.rbs +0 -21
  549. data/sig/hakumi/layout/content/component.rbs +0 -19
  550. data/sig/hakumi/layout/footer/component.rbs +0 -19
  551. data/sig/hakumi/layout/header/component.rbs +0 -19
  552. data/sig/hakumi/layout/sider/component.rbs +0 -19
  553. data/sig/hakumi/mentions/component.rbs +0 -75
  554. data/sig/hakumi/menu/component.rbs +0 -52
  555. data/sig/hakumi/menu/divider/component.rbs +0 -11
  556. data/sig/hakumi/menu/group/component.rbs +0 -20
  557. data/sig/hakumi/menu/item/component.rbs +0 -42
  558. data/sig/hakumi/menu/sub_menu/component.rbs +0 -37
  559. data/sig/hakumi/message/component.rbs +0 -50
  560. data/sig/hakumi/modal/component.rbs +0 -76
  561. data/sig/hakumi/modal/confirm/component.rbs +0 -32
  562. data/sig/hakumi/modal/error/component.rbs +0 -27
  563. data/sig/hakumi/modal/info/component.rbs +0 -27
  564. data/sig/hakumi/modal/success/component.rbs +0 -27
  565. data/sig/hakumi/modal/warning/component.rbs +0 -27
  566. data/sig/hakumi/notification/component.rbs +0 -64
  567. data/sig/hakumi/pagination/component.rbs +0 -113
  568. data/sig/hakumi/popconfirm/component.rbs +0 -52
  569. data/sig/hakumi/popover/component.rbs +0 -66
  570. data/sig/hakumi/progress/component.rbs +0 -201
  571. data/sig/hakumi/qr_code/component.rbs +0 -111
  572. data/sig/hakumi/radio/component.rbs +0 -55
  573. data/sig/hakumi/radio/group/component.rbs +0 -71
  574. data/sig/hakumi/rails/attribute_introspection.rbs +0 -16
  575. data/sig/hakumi/rails/validation_introspection.rbs +0 -18
  576. data/sig/hakumi/rails/validation_mapper.rbs +0 -53
  577. data/sig/hakumi/rails.rbs +0 -6
  578. data/sig/hakumi/rate/component.rbs +0 -56
  579. data/sig/hakumi/result/component.rbs +0 -65
  580. data/sig/hakumi/segmented/component.rbs +0 -72
  581. data/sig/hakumi/select/component.rbs +0 -73
  582. data/sig/hakumi/skeleton/avatar/component.rbs +0 -23
  583. data/sig/hakumi/skeleton/button/component.rbs +0 -24
  584. data/sig/hakumi/skeleton/component.rbs +0 -76
  585. data/sig/hakumi/skeleton/image/component.rbs +0 -22
  586. data/sig/hakumi/skeleton/input/component.rbs +0 -23
  587. data/sig/hakumi/skeleton/node/component.rbs +0 -21
  588. data/sig/hakumi/slider/component.rbs +0 -85
  589. data/sig/hakumi/space/compact/component.rbs +0 -21
  590. data/sig/hakumi/space/component.rbs +0 -47
  591. data/sig/hakumi/spin/component.rbs +0 -77
  592. data/sig/hakumi/splitter/component.rbs +0 -27
  593. data/sig/hakumi/splitter/panel/component.rbs +0 -36
  594. data/sig/hakumi/statistic/component.rbs +0 -97
  595. data/sig/hakumi/steps/component.rbs +0 -82
  596. data/sig/hakumi/steps/item/component.rbs +0 -29
  597. data/sig/hakumi/switch/component.rbs +0 -52
  598. data/sig/hakumi/table/column/component.rbs +0 -57
  599. data/sig/hakumi/table/column_group/component.rbs +0 -20
  600. data/sig/hakumi/table/component.rbs +0 -279
  601. data/sig/hakumi/table/concerns/columns.rbs +0 -95
  602. data/sig/hakumi/table/concerns/editable.rbs +0 -40
  603. data/sig/hakumi/table/concerns/ellipsis.rbs +0 -27
  604. data/sig/hakumi/table/concerns/fixed_columns.rbs +0 -33
  605. data/sig/hakumi/tabs/component.rbs +0 -66
  606. data/sig/hakumi/tabs/item/component.rbs +0 -29
  607. data/sig/hakumi/tag/component.rbs +0 -86
  608. data/sig/hakumi/tag/group/component.rbs +0 -24
  609. data/sig/hakumi/time_picker/component.rbs +0 -72
  610. data/sig/hakumi/timeline/component.rbs +0 -61
  611. data/sig/hakumi/timeline/item/component.rbs +0 -70
  612. data/sig/hakumi/tooltip/component.rbs +0 -73
  613. data/sig/hakumi/tour/component.rbs +0 -84
  614. data/sig/hakumi/transfer/component.rbs +0 -75
  615. data/sig/hakumi/tree/component.rbs +0 -126
  616. data/sig/hakumi/tree_select/component.rbs +0 -122
  617. data/sig/hakumi/typography/base_component.rbs +0 -57
  618. data/sig/hakumi/typography/link/component.rbs +0 -28
  619. data/sig/hakumi/typography/paragraph/component.rbs +0 -13
  620. data/sig/hakumi/typography/title/component.rbs +0 -28
  621. data/sig/hakumi/upload/component.rbs +0 -78
  622. data/sig/hakumi.rbs +0 -96
  623. data/sig/rails/active_model/validations/comparison_validator.rbs +0 -6
  624. data/sig/rails.rbs +0 -18
  625. data/sig/view_component/base.rbs +0 -28
  626. /data/app/components/{hakumi → hakumi_components}/affix/component.html.erb +0 -0
  627. /data/app/components/{hakumi → hakumi_components}/anchor/component.html.erb +0 -0
  628. /data/app/components/{hakumi → hakumi_components}/breadcrumb/component.html.erb +0 -0
  629. /data/app/components/{hakumi → hakumi_components}/dropdown/component.html.erb +0 -0
  630. /data/app/components/{hakumi → hakumi_components}/dropdown/item/component.html.erb +0 -0
  631. /data/app/components/{hakumi → hakumi_components}/icon/component.html.erb +0 -0
  632. /data/app/components/{hakumi → hakumi_components}/menu/component.html.erb +0 -0
  633. /data/app/components/{hakumi → hakumi_components}/menu/group/component.html.erb +0 -0
  634. /data/app/components/{hakumi → hakumi_components}/menu/item/component.html.erb +0 -0
  635. /data/app/components/{hakumi → hakumi_components}/menu/sub_menu/component.html.erb +0 -0
  636. /data/app/components/{hakumi → hakumi_components}/message/component.html.erb +0 -0
  637. /data/app/components/{hakumi → hakumi_components}/qr_code/component.html.erb +0 -0
  638. /data/app/components/{hakumi → hakumi_components}/result/component.html.erb +0 -0
  639. /data/app/components/{hakumi → hakumi_components}/skeleton/avatar/component.html.erb +0 -0
  640. /data/app/components/{hakumi → hakumi_components}/skeleton/button/component.html.erb +0 -0
  641. /data/app/components/{hakumi → hakumi_components}/skeleton/component.html.erb +0 -0
  642. /data/app/components/{hakumi → hakumi_components}/skeleton/input/component.html.erb +0 -0
  643. /data/app/components/{hakumi → hakumi_components}/skeleton/node/component.html.erb +0 -0
  644. /data/app/components/{hakumi → hakumi_components}/spin/component.html.erb +0 -0
  645. /data/app/components/{hakumi → hakumi_components}/statistic/component.html.erb +0 -0
  646. /data/app/components/{hakumi → hakumi_components}/steps/item/component.html.erb +0 -0
  647. /data/app/components/{hakumi → hakumi_components}/tour/component.html.erb +0 -0
@@ -0,0 +1,342 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Tree
6
+ class Component < HakumiComponents::BaseComponent
7
+ extend T::Sig
8
+
9
+ include Concerns::NodeRendering
10
+
11
+ DEFAULT_INDENT = T.let(24, Integer)
12
+ RawNode = T.type_alias { HakumiComponents::Tree::Node::RawInput }
13
+ NodeInput = T.type_alias { T.any(HakumiComponents::Tree::Node, RawNode) }
14
+ NodeKey = T.type_alias { Types::ValidationComparable }
15
+ NodeKeyList = T.type_alias { T::Array[NodeKey] }
16
+ IconInput = T.type_alias { BaseComponent::SymbolInput }
17
+ HeightInput = T.type_alias { T.nilable(BaseComponent::DimensionInput) }
18
+ ControllerLocalValue = T.type_alias { T.nilable(T.any(Types::HtmlPrimitive, T::Array[HakumiComponents::Tree::Node], NodeKeyList)) }
19
+
20
+ class NodeState < T::Struct
21
+ const :expanded, T::Boolean
22
+ const :selected, T::Boolean
23
+ const :checked, T::Boolean
24
+ const :half_checked, T::Boolean
25
+ end
26
+
27
+ sig do
28
+ params(params: BaseComponent::ControllerOptions).returns(T::Hash[Symbol, ControllerLocalValue])
29
+ end
30
+ def self.extract_controller_locals(params)
31
+ raw_nodes = parse_raw_nodes(html_param(params, :nodes) || html_param(params, :tree_data))
32
+
33
+ {
34
+ nodes: HakumiComponents::Tree::Node.coerce_all(raw_nodes),
35
+ checkable: boolean_html_param(html_param(params, :checkable)) == true,
36
+ selectable: boolean_html_param(html_param(params, :selectable)) != false,
37
+ disabled: boolean_html_param(html_param(params, :disabled)) == true,
38
+ default_expand_keys: parse_json_array(html_param(params, :default_expand_keys)),
39
+ default_selected_keys: parse_json_array(html_param(params, :default_selected_keys)),
40
+ default_checked_keys: parse_json_array(html_param(params, :default_checked_keys)),
41
+ expanded_keys: parse_json_array(html_param(params, :expanded_keys), allow_nil: true),
42
+ selected_keys: parse_json_array(html_param(params, :selected_keys), allow_nil: true),
43
+ checked_keys: parse_json_array(html_param(params, :checked_keys), allow_nil: true),
44
+ show_line: boolean_html_param(html_param(params, :show_line)) == true,
45
+ switcher_icon: symbol_html_param(html_param(params, :switcher_icon)),
46
+ icon: symbol_html_param(html_param(params, :icon)),
47
+ directory: boolean_html_param(html_param(params, :directory)) == true,
48
+ multiple: boolean_html_param(html_param(params, :multiple)) == true,
49
+ draggable: boolean_html_param(html_param(params, :draggable)) == true,
50
+ block_node: boolean_html_param(html_param(params, :block_node)) == true,
51
+ searchable: boolean_html_param(html_param(params, :searchable)) == true,
52
+ height: html_primitive_param(html_param(params, :height)),
53
+ check_strictly: boolean_html_param(html_param(params, :check_strictly)) == true
54
+ }
55
+ end
56
+
57
+ sig do
58
+ params(
59
+ nodes: T::Array[NodeInput],
60
+ checkable: T::Boolean,
61
+ selectable: T::Boolean,
62
+ disabled: T::Boolean,
63
+ default_expand_keys: NodeKeyList,
64
+ default_selected_keys: NodeKeyList,
65
+ default_checked_keys: NodeKeyList,
66
+ expanded_keys: T.nilable(NodeKeyList),
67
+ selected_keys: T.nilable(NodeKeyList),
68
+ checked_keys: T.nilable(NodeKeyList),
69
+ show_line: T::Boolean,
70
+ switcher_icon: IconInput,
71
+ icon: IconInput,
72
+ directory: T::Boolean,
73
+ multiple: T::Boolean,
74
+ draggable: T::Boolean,
75
+ block_node: T::Boolean,
76
+ searchable: T::Boolean,
77
+ search_placeholder: T.nilable(String),
78
+ height: HeightInput,
79
+ check_strictly: T::Boolean,
80
+ indent: Integer,
81
+ html_options: Types::HtmlAttributeValue
82
+ ).void
83
+ end
84
+ def initialize(
85
+ nodes:,
86
+ checkable: false,
87
+ selectable: true,
88
+ disabled: false,
89
+ default_expand_keys: [],
90
+ default_selected_keys: [],
91
+ default_checked_keys: [],
92
+ expanded_keys: nil,
93
+ selected_keys: nil,
94
+ checked_keys: nil,
95
+ show_line: false,
96
+ switcher_icon: nil,
97
+ icon: nil,
98
+ directory: false,
99
+ multiple: false,
100
+ draggable: false,
101
+ block_node: false,
102
+ searchable: false,
103
+ search_placeholder: nil,
104
+ height: nil,
105
+ check_strictly: false,
106
+ indent: DEFAULT_INDENT,
107
+ **html_options
108
+ )
109
+ @checkable = T.let(checkable, T::Boolean)
110
+ @selectable = T.let(selectable, T::Boolean)
111
+ @disabled = T.let(disabled, T::Boolean)
112
+ @default_expand_keys = T.let(default_expand_keys.map(&:to_s), T::Array[String])
113
+ @default_selected_keys = T.let(default_selected_keys.map(&:to_s), T::Array[String])
114
+ @default_checked_keys = T.let(default_checked_keys.map(&:to_s), T::Array[String])
115
+ @expanded_keys = T.let(expanded_keys&.map(&:to_s), T.nilable(T::Array[String]))
116
+ @selected_keys = T.let(selected_keys&.map(&:to_s), T.nilable(T::Array[String]))
117
+ @checked_keys = T.let(checked_keys&.map(&:to_s), T.nilable(T::Array[String]))
118
+ @show_line = T.let(show_line, T::Boolean)
119
+ @switcher_icon = T.let(switcher_icon, IconInput)
120
+ @icon = T.let(icon, IconInput)
121
+ @directory = T.let(directory, T::Boolean)
122
+ @multiple = T.let(multiple, T::Boolean)
123
+ @draggable = T.let(draggable, T::Boolean)
124
+ @block_node = T.let(block_node, T::Boolean)
125
+ @searchable = T.let(searchable, T::Boolean)
126
+ @search_placeholder = T.let(search_placeholder || t_default(:search_placeholder, default: "Search"), String)
127
+ @height = T.let(height, HeightInput)
128
+ @check_strictly = T.let(check_strictly, T::Boolean)
129
+ @indent = T.let(indent, Integer)
130
+ @html_options = T.let(html_options, Types::HtmlAttributes)
131
+ @normalized_nodes = T.let(
132
+ HakumiComponents::Tree::Node.coerce_all(
133
+ nodes,
134
+ default_disabled: @disabled,
135
+ default_icon: @icon,
136
+ default_switcher_icon: @switcher_icon
137
+ ),
138
+ T::Array[HakumiComponents::Tree::Node]
139
+ )
140
+
141
+ validate_props!
142
+ end
143
+
144
+ sig { returns(T::Array[HakumiComponents::Tree::Node]) }
145
+ attr_reader :normalized_nodes
146
+
147
+ sig { returns(Types::HtmlAttributes) }
148
+ def wrapper_attributes
149
+ merge_attributes(
150
+ {
151
+ class: wrapper_classes,
152
+ data: data_attributes
153
+ },
154
+ @html_options
155
+ )
156
+ end
157
+
158
+ sig { returns(Types::HtmlAttributes) }
159
+ def list_attributes
160
+ attrs = T.let(
161
+ {
162
+ class: list_classes,
163
+ role: "tree",
164
+ data: { hakumi__tree_target: "list" }
165
+ },
166
+ Types::HtmlAttributes
167
+ )
168
+ attrs[:style] = "max-height: #{@height}px;" if @height
169
+ attrs
170
+ end
171
+
172
+ sig { returns(T::Boolean) }
173
+ def searchable?
174
+ @searchable
175
+ end
176
+
177
+ sig { returns(String) }
178
+ def search_placeholder
179
+ @search_placeholder
180
+ end
181
+
182
+ private
183
+
184
+ sig { returns(String) }
185
+ def wrapper_classes
186
+ component_classes(
187
+ "tree",
188
+ {
189
+ checkable: @checkable,
190
+ "show-line": @show_line,
191
+ directory: @directory,
192
+ draggable: @draggable,
193
+ "block-node": @block_node,
194
+ disabled: @disabled,
195
+ searchable: @searchable,
196
+ virtual: !@height.nil?
197
+ },
198
+ @html_options
199
+ )
200
+ end
201
+
202
+ sig { returns(String) }
203
+ def list_classes
204
+ class_names("tree-list", { virtual: !@height.nil? })
205
+ end
206
+
207
+ sig { returns(Types::DataAttributes) }
208
+ def data_attributes
209
+ data = T.let(
210
+ {
211
+ controller: "hakumi--tree",
212
+ hakumi__tree_checkable_value: @checkable,
213
+ hakumi__tree_selectable_value: @selectable,
214
+ hakumi__tree_disabled_value: @disabled,
215
+ hakumi__tree_default_expand_keys_value: @default_expand_keys.to_json,
216
+ hakumi__tree_default_selected_keys_value: @default_selected_keys.to_json,
217
+ hakumi__tree_default_checked_keys_value: @default_checked_keys.to_json,
218
+ hakumi__tree_show_line_value: @show_line,
219
+ hakumi__tree_switcher_icon_value: @switcher_icon&.to_s,
220
+ hakumi__tree_icon_value: @icon&.to_s,
221
+ hakumi__tree_directory_value: @directory,
222
+ hakumi__tree_multiple_value: @multiple,
223
+ hakumi__tree_draggable_value: @draggable,
224
+ hakumi__tree_block_node_value: @block_node,
225
+ hakumi__tree_searchable_value: @searchable,
226
+ hakumi__tree_height_value: @height,
227
+ hakumi__tree_check_strictly_value: @check_strictly
228
+ },
229
+ Types::DataAttributes
230
+ )
231
+ data[:hakumi__tree_expanded_keys_value] = @expanded_keys.to_json if @expanded_keys
232
+ data[:hakumi__tree_selected_keys_value] = @selected_keys.to_json if @selected_keys
233
+ data[:hakumi__tree_checked_keys_value] = @checked_keys.to_json if @checked_keys
234
+ data
235
+ end
236
+
237
+ sig { void }
238
+ def validate_props!
239
+ raise ArgumentError, "indent must be positive" unless @indent.positive?
240
+ end
241
+
242
+ sig { override.returns(T::Boolean) }
243
+ def tree_checkable?
244
+ @checkable
245
+ end
246
+
247
+ sig { override.returns(T::Boolean) }
248
+ def tree_show_line?
249
+ @show_line
250
+ end
251
+
252
+ sig { override.returns(IconInput) }
253
+ def tree_switcher_icon
254
+ @switcher_icon
255
+ end
256
+
257
+ sig { override.returns(IconInput) }
258
+ def tree_icon
259
+ @icon
260
+ end
261
+
262
+ sig { override.returns(T::Boolean) }
263
+ def tree_directory?
264
+ @directory
265
+ end
266
+
267
+ sig { params(value: Types::HtmlAttributeValue).returns(T::Array[RawNode]) }
268
+ def self.parse_raw_nodes(value)
269
+ return [] if value.nil?
270
+
271
+ array = if value.is_a?(Array)
272
+ value
273
+ elsif value.is_a?(String)
274
+ parsed = JSON.parse(value)
275
+ parsed.is_a?(Array) ? parsed : []
276
+ else
277
+ []
278
+ end
279
+
280
+ array.filter_map do |entry|
281
+ next unless entry.is_a?(Hash)
282
+ typed_entry = T.let({}, RawNode)
283
+ entry.each do |key, item|
284
+ next unless key.is_a?(String) || key.is_a?(Symbol)
285
+ next unless item.nil? || item.is_a?(String) || item.is_a?(Symbol) || item.is_a?(Numeric) || item == true || item == false || item.is_a?(Array) || item.is_a?(Hash)
286
+
287
+ typed_entry[key] = item
288
+ end
289
+ typed_entry
290
+ end
291
+ rescue JSON::ParserError
292
+ []
293
+ end
294
+
295
+ sig { params(value: Types::HtmlAttributeValue, allow_nil: T::Boolean).returns(T.nilable(NodeKeyList)) }
296
+ def self.parse_json_array(value, allow_nil: false)
297
+ return nil if allow_nil && value.nil?
298
+ return [] if value.nil?
299
+
300
+ array = if value.is_a?(Array)
301
+ value
302
+ elsif value.is_a?(String)
303
+ parsed = JSON.parse(value)
304
+ parsed.is_a?(Array) ? parsed : []
305
+ else
306
+ []
307
+ end
308
+
309
+ array.filter_map do |entry|
310
+ case entry
311
+ when String, Symbol, Numeric
312
+ entry
313
+ else
314
+ nil
315
+ end
316
+ end
317
+ rescue JSON::ParserError
318
+ []
319
+ end
320
+
321
+ sig { override.params(key: String).returns(T::Boolean) }
322
+ def expanded?(key)
323
+ (@expanded_keys || @default_expand_keys).include?(key)
324
+ end
325
+
326
+ sig { override.params(key: String).returns(T::Boolean) }
327
+ def selected?(key)
328
+ (@selected_keys || @default_selected_keys).include?(key)
329
+ end
330
+
331
+ sig { override.params(key: String).returns(T::Boolean) }
332
+ def checked?(key)
333
+ (@checked_keys || @default_checked_keys).include?(key)
334
+ end
335
+
336
+ sig { override.params(_key: String).returns(T::Boolean) }
337
+ def half_checked?(_key)
338
+ false
339
+ end
340
+ end
341
+ end
342
+ end
@@ -0,0 +1,365 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module HakumiComponents
5
+ module Tree
6
+ module Concerns
7
+ module NodeRendering
8
+ extend ActiveSupport::Concern
9
+ extend T::Sig
10
+ extend T::Helpers
11
+
12
+ abstract!
13
+
14
+ requires_ancestor { HakumiComponents::BaseComponent }
15
+
16
+ NodeState = T.type_alias { HakumiComponents::Tree::Component::NodeState }
17
+ IconInput = T.type_alias { HakumiComponents::BaseComponent::SymbolInput }
18
+ AriaAttributes = T.type_alias { T::Hash[Symbol, Types::HtmlPrimitive] }
19
+
20
+ sig { returns(ActiveSupport::SafeBuffer) }
21
+ def render_node_template
22
+ switcher_content = render_switcher_icon_markup(resolve_switcher_config(tree_switcher_icon))
23
+ switcher = renderer.content_tag(:button, type: "button", class: "hakumi-tree-switcher", data: { action: "click->hakumi--tree#toggle" }) do
24
+ switcher_content
25
+ end
26
+ checkbox = if tree_checkable?
27
+ renderer.content_tag(:button, type: "button", class: "hakumi-tree-checkbox", data: { action: "click->hakumi--tree#toggleCheck" }) do
28
+ renderer.content_tag(:span, "", class: "hakumi-tree-checkbox-inner")
29
+ end
30
+ end
31
+ icon = renderer.content_tag(:span, default_icon_markup, class: "hakumi-tree-icon", style: default_icon_style)
32
+ title = renderer.content_tag(:button, type: "button", class: "hakumi-tree-title", data: { action: "click->hakumi--tree#select" }) do
33
+ renderer.content_tag(:span, "", class: "hakumi-tree-title-text")
34
+ end
35
+ indent = renderer.content_tag(:span, "", class: "hakumi-tree-indent", data: { hakumi__tree_target: "indent" })
36
+ content_inner = renderer.safe_join([ checkbox, icon, title ].compact)
37
+ content_wrapper = renderer.content_tag(:div, content_inner, class: "hakumi-tree-node-content-wrapper", data: { hakumi__tree_target: "content", action: "click->hakumi--tree#select" })
38
+ row = renderer.content_tag(:div, renderer.safe_join([ indent, switcher, content_wrapper ]), class: "hakumi-tree-node-row")
39
+ children = renderer.content_tag(:div, "", class: "hakumi-tree-children", role: "group", data: { hakumi__tree_target: "children" })
40
+
41
+ renderer.content_tag(:div, renderer.safe_join([ row, children ]), class: "hakumi-tree-node", role: "treeitem", tabindex: "-1", data: { hakumi__tree_target: "node" })
42
+ end
43
+
44
+ sig { params(level_mask: T::Array[T::Boolean]).returns(ActiveSupport::SafeBuffer) }
45
+ def render_indent(level_mask)
46
+ inner_html = level_mask.inject(ActiveSupport::SafeBuffer.new) do |buffer, has_line|
47
+ classes = [ "hakumi-tree-indent-unit" ]
48
+ classes << "hakumi-tree-indent-unit-active" if has_line
49
+ buffer << renderer.content_tag(:span, "", class: classes)
50
+ end
51
+
52
+ renderer.content_tag(:span, inner_html, class: "hakumi-tree-indent", aria: { hidden: true })
53
+ end
54
+
55
+ sig do
56
+ params(
57
+ nodes: T::Array[HakumiComponents::Tree::Node],
58
+ level_mask: T::Array[T::Boolean],
59
+ level: Integer,
60
+ parent_key: T.nilable(String)
61
+ ).returns(ActiveSupport::SafeBuffer)
62
+ end
63
+ def render_nodes(nodes, level_mask: [], level: 0, parent_key: nil)
64
+ renderer.safe_join(nodes.map.with_index do |node, index|
65
+ has_next = index < nodes.size - 1
66
+ render_node(node, index: index, total: nodes.size, level_mask: level_mask, has_next: has_next, level: level, parent_key: parent_key)
67
+ end)
68
+ end
69
+
70
+ private
71
+
72
+ sig { abstract.returns(T::Boolean) }
73
+ def tree_checkable?; end
74
+
75
+ sig { abstract.returns(T::Boolean) }
76
+ def tree_show_line?; end
77
+
78
+ sig { abstract.returns(IconInput) }
79
+ def tree_switcher_icon; end
80
+
81
+ sig { abstract.returns(IconInput) }
82
+ def tree_icon; end
83
+
84
+ sig { abstract.returns(T::Boolean) }
85
+ def tree_directory?; end
86
+
87
+ sig { abstract.params(key: String).returns(T::Boolean) }
88
+ def expanded?(key); end
89
+
90
+ sig { abstract.params(key: String).returns(T::Boolean) }
91
+ def selected?(key); end
92
+
93
+ sig { abstract.params(key: String).returns(T::Boolean) }
94
+ def checked?(key); end
95
+
96
+ sig { abstract.params(key: String).returns(T::Boolean) }
97
+ def half_checked?(key); end
98
+
99
+ sig { returns(T.untyped) }
100
+ def renderer
101
+ T.unsafe(self)
102
+ end
103
+
104
+ sig do
105
+ params(
106
+ node: HakumiComponents::Tree::Node,
107
+ index: Integer,
108
+ total: Integer,
109
+ level_mask: T::Array[T::Boolean],
110
+ has_next: T::Boolean,
111
+ level: Integer,
112
+ parent_key: T.nilable(String)
113
+ ).returns(ActiveSupport::SafeBuffer)
114
+ end
115
+ def render_node(node, index:, total:, level_mask:, has_next:, level:, parent_key:)
116
+ state = node_state(node)
117
+
118
+ attrs = node_attributes(
119
+ node,
120
+ state,
121
+ index: index,
122
+ total: total,
123
+ has_next: has_next,
124
+ level: level,
125
+ parent_key: parent_key
126
+ )
127
+
128
+ renderer.content_tag(:div, attrs) do
129
+ renderer.safe_join([
130
+ render_node_row(node, state, level_mask),
131
+ render_node_children(node, level_mask: level_mask + [ has_next ], level: level + 1)
132
+ ].compact)
133
+ end
134
+ end
135
+
136
+ sig do
137
+ params(
138
+ node: HakumiComponents::Tree::Node,
139
+ state: NodeState,
140
+ index: Integer,
141
+ total: Integer,
142
+ has_next: T::Boolean,
143
+ level: Integer,
144
+ parent_key: T.nilable(String)
145
+ ).returns(Types::HtmlAttributes)
146
+ end
147
+ def node_attributes(node, state, index:, total:, has_next:, level:, parent_key:)
148
+ T.let(
149
+ {
150
+ class: node_classes(node, state),
151
+ role: "treeitem",
152
+ tabindex: -1,
153
+ data: node_data(node, index: index, total: total, has_next: has_next, level: level, parent_key: parent_key),
154
+ aria: node_aria(node, state)
155
+ },
156
+ Types::HtmlAttributes
157
+ )
158
+ end
159
+
160
+ sig do
161
+ params(
162
+ node: HakumiComponents::Tree::Node,
163
+ state: NodeState,
164
+ level_mask: T::Array[T::Boolean]
165
+ ).returns(ActiveSupport::SafeBuffer)
166
+ end
167
+ def render_node_row(node, state, level_mask)
168
+ renderer.content_tag(:div, class: "hakumi-tree-node-row") do
169
+ renderer.safe_join([
170
+ render_indent(level_mask),
171
+ render_switcher(node),
172
+ render_node_content_wrapper(node, state)
173
+ ].compact)
174
+ end
175
+ end
176
+
177
+ sig { params(node: HakumiComponents::Tree::Node).returns(NodeState) }
178
+ def node_state(node)
179
+ HakumiComponents::Tree::Component::NodeState.new(
180
+ expanded: expanded?(node.key),
181
+ selected: selected?(node.key),
182
+ checked: checked?(node.key),
183
+ half_checked: half_checked?(node.key)
184
+ )
185
+ end
186
+
187
+ sig { params(node: HakumiComponents::Tree::Node, state: NodeState).returns(String) }
188
+ def node_classes(node, state)
189
+ renderer.class_names(
190
+ "tree-node",
191
+ {
192
+ disabled: node.disabled,
193
+ selected: state.selected,
194
+ checked: state.checked,
195
+ "half-checked": state.half_checked,
196
+ expanded: state.expanded,
197
+ leaf: node.leaf,
198
+ async: node.async,
199
+ loading: node.loading
200
+ }
201
+ )
202
+ end
203
+
204
+ sig do
205
+ params(
206
+ node: HakumiComponents::Tree::Node,
207
+ index: Integer,
208
+ total: Integer,
209
+ has_next: T::Boolean,
210
+ level: Integer,
211
+ parent_key: T.nilable(String)
212
+ ).returns(Types::DataAttributes)
213
+ end
214
+ def node_data(node, index:, total:, has_next:, level:, parent_key:)
215
+ {
216
+ hakumi__tree_target: "node",
217
+ key: node.key,
218
+ parent_key: parent_key,
219
+ level: level,
220
+ title: node.title,
221
+ selectable: node.selectable,
222
+ checkable: node.checkable,
223
+ disabled: node.disabled,
224
+ leaf: node.leaf,
225
+ async: node.async,
226
+ loading: node.loading,
227
+ index: index,
228
+ total: total,
229
+ has_next: has_next
230
+ }
231
+ end
232
+
233
+ sig { params(node: HakumiComponents::Tree::Node, state: NodeState).returns(AriaAttributes) }
234
+ def node_aria(node, state)
235
+ aria = T.let({ selected: state.selected }, AriaAttributes)
236
+ aria[:expanded] = state.expanded unless node.leaf
237
+ aria
238
+ end
239
+
240
+ sig { params(node: HakumiComponents::Tree::Node, state: NodeState).returns(ActiveSupport::SafeBuffer) }
241
+ def render_node_content_wrapper(node, state)
242
+ renderer.content_tag(:div, class: "hakumi-tree-node-content-wrapper", data: { hakumi__tree_target: "content", action: "click->hakumi--tree#select" }) do
243
+ renderer.safe_join([ render_checkbox(node, state), render_icon(node), render_title(node) ].compact)
244
+ end
245
+ end
246
+
247
+ sig { params(node: HakumiComponents::Tree::Node).returns(ActiveSupport::SafeBuffer) }
248
+ def render_switcher(node)
249
+ switcher_conf = resolve_switcher_config(node.switcher_icon)
250
+ renderer.content_tag(:button,
251
+ type: "button",
252
+ class: switcher_classes(node.leaf),
253
+ aria: { label: node.leaf ? "Leaf" : "Toggle" },
254
+ data: { action: (node.leaf ? nil : "click->hakumi--tree#toggle"), hakumi__tree_target: "switcher", key: node.key }.compact
255
+ ) do
256
+ if node.leaf
257
+ renderer.content_tag(:span, "", class: "hakumi-tree-switcher-noop")
258
+ else
259
+ render_switcher_icon_markup(switcher_conf)
260
+ end
261
+ end
262
+ end
263
+
264
+ sig { params(icon_or_conf: IconInput).returns(T::Hash[Symbol, Symbol]) }
265
+ def resolve_switcher_config(icon_or_conf)
266
+ candidate = icon_or_conf || tree_switcher_icon
267
+ if candidate
268
+ case candidate.to_sym
269
+ when :plus, :plus_square, :plus_circle
270
+ suffix = candidate.to_s.gsub("plus", "")
271
+ { collapse: candidate.to_sym, expand: "minus#{suffix}".to_sym }
272
+ when :folder, :folder_open
273
+ { collapse: :folder, expand: :folder_open }
274
+ else
275
+ { default: candidate.to_sym }
276
+ end
277
+ elsif tree_show_line?
278
+ { collapse: :plus_square, expand: :minus_square }
279
+ else
280
+ { default: :caret_right }
281
+ end
282
+ end
283
+
284
+ sig { params(node: HakumiComponents::Tree::Node, state: NodeState).returns(T.nilable(ActiveSupport::SafeBuffer)) }
285
+ def render_checkbox(node, state)
286
+ return nil unless tree_checkable? && node.checkable
287
+
288
+ renderer.content_tag(:button,
289
+ type: "button",
290
+ class: "hakumi-tree-checkbox",
291
+ aria: { checked: state.checked },
292
+ data: { action: "click->hakumi--tree#toggleCheck", hakumi__tree_target: "checkbox", key: node.key, disabled: node.disable_checkbox }
293
+ ) do
294
+ renderer.content_tag(:span, "", class: "hakumi-tree-checkbox-inner")
295
+ end
296
+ end
297
+
298
+ sig { params(node: HakumiComponents::Tree::Node).returns(T.nilable(ActiveSupport::SafeBuffer)) }
299
+ def render_icon(node)
300
+ icon_name = node.icon
301
+ icon_name = node.leaf ? :file : :folder if icon_name.nil? && tree_directory?
302
+ return nil if icon_name.nil?
303
+
304
+ renderer.content_tag(:span, class: "hakumi-tree-icon") do
305
+ renderer.render HakumiComponents::Icon::Component.new(name: icon_name)
306
+ end
307
+ end
308
+
309
+ sig { params(node: HakumiComponents::Tree::Node).returns(ActiveSupport::SafeBuffer) }
310
+ def render_title(node)
311
+ renderer.content_tag(:button,
312
+ type: "button",
313
+ class: "hakumi-tree-title",
314
+ data: { action: "click->hakumi--tree#select", hakumi__tree_target: "title", key: node.key }
315
+ ) do
316
+ renderer.content_tag(:span, node.title, class: "hakumi-tree-title-text", data: { title: node.title })
317
+ end
318
+ end
319
+
320
+ sig { params(node: HakumiComponents::Tree::Node, level_mask: T::Array[T::Boolean], level: Integer).returns(T.nilable(ActiveSupport::SafeBuffer)) }
321
+ def render_node_children(node, level_mask: [], level: 0)
322
+ return nil if node.children.empty?
323
+
324
+ renderer.content_tag(:div, class: "hakumi-tree-children", role: "group", data: { hakumi__tree_target: "children" }) do
325
+ render_nodes(node.children, level_mask: level_mask, level: level, parent_key: node.key)
326
+ end
327
+ end
328
+
329
+ sig { params(is_leaf: T::Boolean).returns(String) }
330
+ def switcher_classes(is_leaf)
331
+ renderer.class_names("tree-switcher", { leaf: is_leaf })
332
+ end
333
+
334
+ sig { params(config: T::Hash[Symbol, Symbol]).returns(ActiveSupport::SafeBuffer) }
335
+ def render_switcher_icon_markup(config)
336
+ if config[:default]
337
+ renderer.content_tag(:span, class: "hakumi-tree-switcher-icon") do
338
+ renderer.render HakumiComponents::Icon::Component.new(name: config[:default])
339
+ end
340
+ else
341
+ renderer.content_tag(:span, class: "hakumi-tree-switcher-icon hakumi-tree-switcher-dual") do
342
+ renderer.safe_join([
343
+ renderer.content_tag(:span, class: "hakumi-tree-icon-collapse") { renderer.render HakumiComponents::Icon::Component.new(name: config.fetch(:collapse)) },
344
+ renderer.content_tag(:span, class: "hakumi-tree-icon-expand") { renderer.render HakumiComponents::Icon::Component.new(name: config.fetch(:expand)) }
345
+ ])
346
+ end
347
+ end
348
+ end
349
+
350
+ sig { returns(T.nilable(ActiveSupport::SafeBuffer)) }
351
+ def default_icon_markup
352
+ icon = tree_icon
353
+ return nil if icon.nil?
354
+
355
+ renderer.render HakumiComponents::Icon::Component.new(name: icon)
356
+ end
357
+
358
+ sig { returns(T.nilable(String)) }
359
+ def default_icon_style
360
+ tree_icon.nil? ? "display: none" : nil
361
+ end
362
+ end
363
+ end
364
+ end
365
+ end