avo 3.30.4 → 4.0.0.beta.2

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 (470) hide show
  1. checksums.yaml +4 -4
  2. data/Gemfile +18 -1
  3. data/Gemfile.lock +257 -174
  4. data/README.md +1 -1
  5. data/app/assets/builds/avo/application.css +13979 -0
  6. data/app/assets/builds/avo/application.js +1160 -0
  7. data/app/assets/builds/avo/application.js.map +7 -0
  8. data/app/assets/builds/avo/late-registration.js +2 -0
  9. data/app/assets/builds/avo/late-registration.js.map +7 -0
  10. data/app/assets/config/avo_manifest.js +2 -1
  11. data/app/assets/images/avo/placeholder.svg +1 -0
  12. data/app/assets/stylesheets/application.css +250 -0
  13. data/app/assets/stylesheets/css/button-group.css +23 -0
  14. data/app/assets/stylesheets/css/components/avatar.css +128 -0
  15. data/app/assets/stylesheets/css/components/breadcrumbs.css +43 -0
  16. data/app/assets/stylesheets/css/components/button.css +343 -0
  17. data/app/assets/stylesheets/css/components/color_scheme_switcher.css +226 -0
  18. data/app/assets/stylesheets/css/components/discreet_information.css +49 -0
  19. data/app/assets/stylesheets/css/components/field-wrapper.css +107 -0
  20. data/app/assets/stylesheets/css/components/grid.css +120 -0
  21. data/app/assets/stylesheets/css/components/input.css +312 -0
  22. data/app/assets/stylesheets/css/components/modal.css +228 -0
  23. data/app/assets/stylesheets/css/components/tooltip.css +25 -0
  24. data/app/assets/stylesheets/css/components/ui/badge.css +143 -0
  25. data/app/assets/stylesheets/css/components/ui/card.css +95 -0
  26. data/app/assets/stylesheets/css/components/ui/checkbox.css +50 -0
  27. data/app/assets/stylesheets/css/components/ui/description_list.css +3 -0
  28. data/app/assets/stylesheets/css/components/ui/dropdown.css +68 -0
  29. data/app/assets/stylesheets/css/components/ui/file_upload_input.css +94 -0
  30. data/app/assets/stylesheets/css/components/ui/file_upload_item.css +78 -0
  31. data/app/assets/stylesheets/css/components/ui/panel.css +59 -0
  32. data/app/assets/stylesheets/css/components/ui/panel_header.css +48 -0
  33. data/app/assets/stylesheets/css/components/ui/radio.css +22 -0
  34. data/app/assets/stylesheets/css/components/ui/tabs.css +74 -0
  35. data/app/assets/stylesheets/css/css-animations.css +54 -0
  36. data/app/assets/stylesheets/css/fields/code.css +75 -9
  37. data/app/assets/stylesheets/css/fields/easy-mde.css +7 -0
  38. data/app/assets/stylesheets/css/fields/key_value.css +97 -0
  39. data/app/assets/stylesheets/css/fields/progress.css +4 -4
  40. data/app/assets/stylesheets/css/fields/status.css +7 -1
  41. data/app/assets/stylesheets/css/fields/tags.css +33 -16
  42. data/app/assets/stylesheets/css/fields/tiptap.css +17 -15
  43. data/app/assets/stylesheets/css/fields/trix.css +62 -2
  44. data/app/assets/stylesheets/css/fonts.css +24 -24
  45. data/app/assets/stylesheets/css/layout.css +135 -0
  46. data/app/assets/stylesheets/css/pagination.css +114 -78
  47. data/app/assets/stylesheets/css/resource-controls.css +13 -0
  48. data/app/assets/stylesheets/css/search.css +22 -12
  49. data/app/assets/stylesheets/css/sidebar.css +310 -24
  50. data/app/assets/stylesheets/css/table.css +60 -0
  51. data/app/assets/stylesheets/css/tooltips.css +1 -1
  52. data/app/assets/stylesheets/css/typography.css +10 -1
  53. data/app/assets/stylesheets/css/variables.css +318 -0
  54. data/app/assets/svgs/avo/moon-plus-plus.svg +1 -0
  55. data/app/components/avo/actions_component.html.erb +30 -36
  56. data/app/components/avo/actions_component.rb +8 -11
  57. data/app/components/avo/alert_component.html.erb +3 -3
  58. data/app/components/avo/alert_component.rb +1 -1
  59. data/app/components/avo/asset_manager/javascript_component.html.erb +1 -2
  60. data/app/components/avo/backtrace_alert_component.html.erb +4 -4
  61. data/app/components/avo/base_component.rb +3 -4
  62. data/app/components/avo/breadcrumb_element_component.html.erb +17 -0
  63. data/app/components/avo/breadcrumb_element_component.rb +21 -0
  64. data/app/components/avo/breadcrumbs_component.html.erb +19 -0
  65. data/app/components/avo/breadcrumbs_component.rb +5 -0
  66. data/app/components/avo/button_component.rb +20 -126
  67. data/app/components/avo/clipboard_component.html.erb +3 -2
  68. data/app/components/avo/clipboard_component.rb +1 -1
  69. data/app/components/avo/component_missing_component.rb +11 -0
  70. data/app/components/avo/cover_component.html.erb +3 -0
  71. data/app/components/avo/cover_component.rb +25 -0
  72. data/app/components/avo/debug/status_component.html.erb +59 -0
  73. data/app/components/avo/debug/status_component.rb +30 -0
  74. data/app/components/avo/description_list_component.rb +11 -0
  75. data/app/components/avo/discreet_information_component.html.erb +31 -6
  76. data/app/components/avo/discreet_information_component.rb +23 -32
  77. data/app/components/avo/divider_component.html.erb +2 -2
  78. data/app/components/avo/empty_state_component.html.erb +2 -9
  79. data/app/components/avo/empty_state_component.rb +0 -8
  80. data/app/components/avo/field_wrapper_component.html.erb +21 -22
  81. data/app/components/avo/field_wrapper_component.rb +9 -18
  82. data/app/components/avo/fields/avatar_field/index_component.html.erb +9 -0
  83. data/app/components/avo/fields/avatar_field/index_component.rb +4 -0
  84. data/app/components/avo/fields/badge_field/index_component.html.erb +7 -1
  85. data/app/components/avo/fields/badge_field/show_component.html.erb +6 -1
  86. data/app/components/avo/fields/belongs_to_field/edit_component.html.erb +16 -14
  87. data/app/components/avo/fields/belongs_to_field/edit_component.rb +13 -3
  88. data/app/components/avo/fields/boolean_field/edit_component.html.erb +2 -2
  89. data/app/components/avo/fields/boolean_group_field/edit_component.html.erb +1 -1
  90. data/app/components/avo/fields/boolean_group_field/edit_component.rb +2 -3
  91. data/app/components/avo/fields/code_field/show_component.html.erb +1 -1
  92. data/app/components/avo/fields/common/boolean_check_component.rb +3 -3
  93. data/app/components/avo/fields/common/boolean_group_component.html.erb +2 -2
  94. data/app/components/avo/fields/common/files/controls_component.html.erb +30 -27
  95. data/app/components/avo/fields/common/files/controls_component.rb +0 -1
  96. data/app/components/avo/fields/common/files/list_viewer_component.html.erb +4 -5
  97. data/app/components/avo/fields/common/files/view_type/grid_item_component.html.erb +1 -1
  98. data/app/components/avo/fields/common/files/view_type/list_item_component.html.erb +7 -19
  99. data/app/components/avo/fields/common/heading_component.html.erb +2 -2
  100. data/app/components/avo/fields/common/heading_component.rb +1 -1
  101. data/app/components/avo/fields/common/key_value_component.html.erb +20 -24
  102. data/app/components/avo/fields/common/nested_field_component.html.erb +1 -1
  103. data/app/components/avo/fields/common/nested_field_component.rb +1 -1
  104. data/app/components/avo/fields/common/progress_bar_component.html.erb +2 -2
  105. data/app/components/avo/fields/common/stars_component.html.erb +1 -1
  106. data/app/components/avo/fields/common/status_viewer_component.html.erb +4 -10
  107. data/app/components/avo/fields/date_field/edit_component.html.erb +14 -1
  108. data/app/components/avo/fields/date_time_field/edit_component.html.erb +3 -3
  109. data/app/components/avo/fields/easy_mde_field/show_component.html.erb +1 -1
  110. data/app/components/avo/fields/edit_component.rb +9 -5
  111. data/app/components/avo/fields/file_field/edit_component.html.erb +14 -6
  112. data/app/components/avo/fields/files_field/edit_component.html.erb +11 -4
  113. data/app/components/avo/fields/has_one_field/show_component.html.erb +10 -9
  114. data/app/components/avo/fields/heading_field/edit_component.html.erb +1 -1
  115. data/app/components/avo/fields/heading_field/show_component.html.erb +2 -2
  116. data/app/components/avo/fields/password_field/edit_component.html.erb +5 -7
  117. data/app/components/avo/fields/progress_bar_field/edit_component.html.erb +1 -1
  118. data/app/components/avo/fields/radio_field/edit_component.html.erb +1 -1
  119. data/app/components/avo/fields/show_component.rb +4 -7
  120. data/app/components/avo/fields/stars_field/edit_component.html.erb +3 -4
  121. data/app/components/avo/fields/tags_field/edit_component.html.erb +1 -1
  122. data/app/components/avo/fields/tags_field/tag_component.html.erb +7 -8
  123. data/app/components/avo/fields/time_field/edit_component.html.erb +3 -3
  124. data/app/components/avo/fields/tiptap_field/edit_component.html.erb +11 -11
  125. data/app/components/avo/fields/tiptap_field/show_component.html.erb +4 -4
  126. data/app/components/avo/fields/trix_field/edit_component.html.erb +2 -2
  127. data/app/components/avo/filters_component.html.erb +8 -16
  128. data/app/components/avo/index/field_wrapper_component.html.erb +9 -2
  129. data/app/components/avo/index/field_wrapper_component.rb +0 -16
  130. data/app/components/avo/index/grid_cover_empty_state_component.html.erb +16 -2
  131. data/app/components/avo/index/grid_cover_empty_state_component.rb +90 -0
  132. data/app/components/avo/index/grid_item_component.html.erb +29 -12
  133. data/app/components/avo/index/grid_item_component.rb +12 -10
  134. data/app/components/avo/index/resource_controls_component.html.erb +1 -1
  135. data/app/components/avo/index/resource_controls_component.rb +7 -17
  136. data/app/components/avo/index/resource_controls_dropdown_component.html.erb +3 -0
  137. data/app/components/avo/index/resource_controls_dropdown_component.rb +81 -0
  138. data/app/components/avo/index/table_row_component.html.erb +11 -11
  139. data/app/components/avo/items/panel_component.html.erb +8 -18
  140. data/app/components/avo/items/panel_component.rb +0 -20
  141. data/app/components/avo/items/switcher_component.html.erb +60 -1
  142. data/app/components/avo/items/switcher_component.rb +10 -5
  143. data/app/components/avo/items/visible_items_component.html.erb +2 -1
  144. data/app/components/avo/items/visible_items_component.rb +1 -0
  145. data/app/components/avo/media_library/item_details_component.html.erb +2 -2
  146. data/app/components/avo/media_library/list_component.html.erb +24 -19
  147. data/app/components/avo/media_library/list_component.rb +2 -2
  148. data/app/components/avo/media_library/list_item_component.html.erb +3 -3
  149. data/app/components/avo/modal_component.html.erb +52 -20
  150. data/app/components/avo/modal_component.rb +7 -12
  151. data/app/components/avo/paginator_component.html.erb +46 -33
  152. data/app/components/avo/paginator_component.rb +10 -5
  153. data/app/components/avo/panel_name_component.html.erb +1 -1
  154. data/app/components/avo/profile_item_component.html.erb +4 -4
  155. data/app/components/avo/profile_item_component.rb +2 -4
  156. data/app/components/avo/referrer_params_component.html.erb +1 -1
  157. data/app/components/avo/resource_component.rb +8 -19
  158. data/app/components/avo/resource_listing_component.html.erb +22 -0
  159. data/app/components/avo/resource_listing_component.rb +28 -0
  160. data/app/components/avo/resource_sidebar_component.html.erb +2 -2
  161. data/app/components/avo/resource_sidebar_component.rb +0 -4
  162. data/app/components/avo/row_component.html.erb +1 -1
  163. data/app/components/avo/row_selector_component.html.erb +2 -4
  164. data/app/components/avo/row_selector_component.rb +0 -1
  165. data/app/components/avo/search_overlay_component.html.erb +6 -0
  166. data/app/components/avo/search_overlay_component.rb +2 -0
  167. data/app/components/avo/sidebar/base_item_component.rb +9 -9
  168. data/app/components/avo/sidebar/group_component.html.erb +23 -27
  169. data/app/components/avo/sidebar/group_component.rb +4 -0
  170. data/app/components/avo/sidebar/link_component.html.erb +30 -7
  171. data/app/components/avo/sidebar/link_component.rb +30 -2
  172. data/app/components/avo/sidebar/section_component.html.erb +31 -11
  173. data/app/components/avo/sidebar/section_component.rb +1 -3
  174. data/app/components/avo/sidebar_component.html.erb +35 -36
  175. data/app/components/avo/sidebar_component.rb +16 -0
  176. data/app/components/avo/sidebar_profile_component.html.erb +27 -36
  177. data/app/components/avo/sidebar_profile_component.rb +14 -0
  178. data/app/components/avo/tab_group_component.html.erb +21 -9
  179. data/app/components/avo/tab_group_component.rb +54 -15
  180. data/app/components/avo/turbo_frame_wrapper_component.html.erb +4 -8
  181. data/app/components/avo/turbo_frame_wrapper_component.rb +9 -0
  182. data/app/components/avo/u_i/avatar_component.html.erb +11 -0
  183. data/app/components/avo/u_i/avatar_component.rb +61 -0
  184. data/app/components/avo/u_i/badge_component.html.erb +9 -0
  185. data/app/components/avo/u_i/badge_component.rb +35 -0
  186. data/app/components/avo/u_i/card_component.html.erb +29 -0
  187. data/app/components/avo/u_i/card_component.rb +65 -0
  188. data/app/components/avo/u_i/dropdown_component.html.erb +14 -0
  189. data/app/components/avo/u_i/dropdown_component.rb +26 -0
  190. data/app/components/avo/u_i/dropdown_menu_component.html.erb +16 -0
  191. data/app/components/avo/u_i/dropdown_menu_component.rb +8 -0
  192. data/app/components/avo/u_i/file_upload_input_component.html.erb +26 -0
  193. data/app/components/avo/u_i/file_upload_input_component.rb +15 -0
  194. data/app/components/avo/u_i/file_upload_item_component.html.erb +48 -0
  195. data/app/components/avo/u_i/file_upload_item_component.rb +17 -0
  196. data/app/components/avo/u_i/icon_button_component.html.erb +7 -0
  197. data/app/components/avo/u_i/icon_button_component.rb +22 -0
  198. data/app/components/avo/u_i/panel_component.html.erb +49 -0
  199. data/app/components/avo/u_i/panel_component.rb +29 -0
  200. data/app/components/avo/u_i/panel_header_component.html.erb +37 -0
  201. data/app/components/avo/u_i/panel_header_component.rb +24 -0
  202. data/app/components/avo/u_i/search_input_component.html.erb +28 -0
  203. data/app/components/avo/u_i/search_input_component.rb +12 -0
  204. data/app/components/avo/u_i/tabs/tab_component.html.erb +26 -0
  205. data/app/components/avo/u_i/tabs/tab_component.rb +25 -0
  206. data/app/components/avo/u_i/tabs/tabs_component.html.erb +7 -0
  207. data/app/components/avo/u_i/tabs/tabs_component.rb +13 -0
  208. data/app/components/avo/view_types/base_view_type_component.rb +26 -0
  209. data/app/components/avo/view_types/grid_component.html.erb +18 -0
  210. data/app/components/avo/view_types/grid_component.rb +4 -0
  211. data/app/components/avo/view_types/map_component.html.erb +17 -0
  212. data/app/components/avo/view_types/map_component.rb +110 -0
  213. data/app/components/avo/view_types/table_component.html.erb +63 -0
  214. data/app/components/avo/{index/resource_table_component.rb → view_types/table_component.rb} +1 -10
  215. data/app/components/avo/views/resource_edit_component.html.erb +12 -18
  216. data/app/components/avo/views/resource_edit_component.rb +1 -27
  217. data/app/components/avo/views/resource_index_component.html.erb +97 -74
  218. data/app/components/avo/views/resource_index_component.rb +23 -3
  219. data/app/components/avo/views/resource_show_component.html.erb +4 -2
  220. data/app/components/avo/views/resource_show_component.rb +1 -4
  221. data/app/controllers/avo/associations_controller.rb +1 -1
  222. data/app/controllers/avo/base_application_controller.rb +7 -5
  223. data/app/controllers/avo/base_controller.rb +139 -40
  224. data/app/controllers/avo/debug_controller.rb +0 -29
  225. data/app/controllers/avo/media_library_controller.rb +17 -1
  226. data/app/controllers/avo/search_controller.rb +5 -0
  227. data/app/controllers/concerns/avo/initializes_avo.rb +7 -8
  228. data/app/helpers/avo/application_helper.rb +60 -46
  229. data/app/helpers/avo/turbo_stream_actions_helper.rb +10 -1
  230. data/app/helpers/avo/url_helpers.rb +3 -2
  231. data/app/javascript/{avo.base.js → application.js} +9 -18
  232. data/app/javascript/js/application.js +40 -0
  233. data/app/javascript/js/controllers/action_controller.js +4 -4
  234. data/app/javascript/js/controllers/actions_overflow_controller.js +21 -6
  235. data/app/javascript/js/controllers/color_scheme_switcher_controller.js +226 -0
  236. data/app/javascript/js/controllers/dropdown_menu_controller.js +42 -0
  237. data/app/javascript/js/controllers/fields/code_field_controller.js +20 -3
  238. data/app/javascript/js/controllers/fields/easy_mde_controller.js +23 -1
  239. data/app/javascript/js/controllers/fields/key_value_controller.js +43 -39
  240. data/app/javascript/js/controllers/fields/panel_refresh_controller.js +4 -0
  241. data/app/javascript/js/controllers/fields/tags_field_controller.js +2 -2
  242. data/app/javascript/js/controllers/fields/tags_field_helpers.js +4 -6
  243. data/app/javascript/js/controllers/grid_cover_empty_state_controller.js +42 -0
  244. data/app/javascript/js/controllers/item_selector_controller.js +0 -2
  245. data/app/javascript/js/controllers/loading_button_controller.js +1 -5
  246. data/app/javascript/js/controllers/map_dark_mode_controller.js +131 -0
  247. data/app/javascript/js/controllers/menu_controller.js +38 -16
  248. data/app/javascript/js/controllers/modal_controller.js +16 -0
  249. data/app/javascript/js/controllers/modal_size_controller.js +83 -0
  250. data/app/javascript/js/controllers/nested_form_controller.js +2 -2
  251. data/app/javascript/js/controllers/password_visibility_controller.js +13 -0
  252. data/app/javascript/js/controllers/preview_controller.js +2 -2
  253. data/app/javascript/js/controllers/resource_search_controller.js +123 -0
  254. data/app/javascript/js/controllers/search_controller.js +10 -29
  255. data/app/javascript/js/controllers/sidebar_controller.js +29 -9
  256. data/app/javascript/js/controllers/table_row_controller.js +28 -0
  257. data/app/javascript/js/controllers/tippy_controller.js +1 -1
  258. data/app/javascript/js/controllers/toggle_controller.js +40 -5
  259. data/app/javascript/js/controllers.js +14 -2
  260. data/app/javascript/js/custom-stream-actions.js +10 -8
  261. data/app/views/avo/actions/show.html.erb +14 -10
  262. data/app/views/avo/base/_boolean_filter.html.erb +1 -1
  263. data/app/views/avo/base/_date_time_filter.html.erb +3 -3
  264. data/app/views/avo/base/_multiple_select_filter.html.erb +1 -1
  265. data/app/views/avo/base/_new_via_belongs_to.html.erb +2 -3
  266. data/app/views/avo/base/_text_filter.html.erb +1 -1
  267. data/app/views/avo/base/preview.html.erb +5 -4
  268. data/app/views/avo/debug/status.html.erb +5 -92
  269. data/app/views/avo/debug/status.text.erb +0 -2
  270. data/app/views/avo/home/_actions.html.erb +1 -1
  271. data/app/views/avo/home/_dashboards.html.erb +1 -1
  272. data/app/views/avo/home/_filters.html.erb +1 -1
  273. data/app/views/avo/home/_resources.html.erb +2 -2
  274. data/app/views/avo/home/failed_to_load.html.erb +2 -2
  275. data/app/views/avo/home/index.html.erb +25 -25
  276. data/app/views/avo/media_library/_form.html.erb +43 -28
  277. data/app/views/avo/media_library/show.html.erb +7 -3
  278. data/app/views/avo/modal/_size_selector.html.erb +46 -0
  279. data/app/views/avo/partials/_alerts.html.erb +1 -1
  280. data/app/views/avo/partials/_color_scheme_switcher.html.erb +106 -0
  281. data/app/views/avo/partials/_color_theme_override.html.erb +49 -0
  282. data/app/views/avo/partials/_confirm_dialog.html.erb +19 -17
  283. data/app/views/avo/partials/_custom_tools_alert.html.erb +6 -6
  284. data/app/views/avo/partials/_footer.html.erb +1 -1
  285. data/app/views/avo/partials/_header.html.erb +1 -1
  286. data/app/views/avo/partials/_javascript.html.erb +0 -2
  287. data/app/views/avo/partials/_navbar.html.erb +53 -8
  288. data/app/views/avo/partials/_sortable_component.html.erb +1 -1
  289. data/app/views/avo/partials/_table_header.html.erb +28 -19
  290. data/app/views/avo/partials/_view_toggle_button.html.erb +6 -29
  291. data/app/views/avo/partials/distribution_chart.html.erb +2 -2
  292. data/app/views/avo/private/_links_and_buttons.html.erb +12 -8
  293. data/app/views/avo/private/design.html.erb +8 -4
  294. data/app/views/avo/sidebar/_license_warning.html.erb +3 -3
  295. data/app/views/layouts/avo/application.html.erb +39 -17
  296. data/app/views/layouts/avo/modal.html.erb +7 -0
  297. data/avo.gemspec +3 -4
  298. data/config/i18n-tasks.yml +1 -1
  299. data/config/importmap.rb +1 -0
  300. data/config/initializers/pagy.rb +5 -25
  301. data/config/routes/dynamic_routes.rb +4 -0
  302. data/config/routes.rb +6 -7
  303. data/db/factories.rb +14 -0
  304. data/lib/avo/asset_manager.rb +2 -0
  305. data/lib/avo/avatar.rb +7 -0
  306. data/lib/avo/base_action.rb +16 -4
  307. data/lib/avo/concerns/breadcrumbs.rb +7 -66
  308. data/lib/avo/concerns/form_builder.rb +41 -0
  309. data/lib/avo/concerns/{has_profile_photo.rb → has_avatar.rb} +8 -4
  310. data/lib/avo/concerns/{has_cover_photo.rb → has_cover.rb} +4 -4
  311. data/lib/avo/concerns/has_description.rb +9 -0
  312. data/lib/avo/concerns/has_item_type.rb +6 -2
  313. data/lib/avo/concerns/has_items.rb +43 -58
  314. data/lib/avo/concerns/pagination.rb +29 -19
  315. data/lib/avo/concerns/row_controls_configuration.rb +22 -15
  316. data/lib/avo/configuration/branding.rb +7 -7
  317. data/lib/avo/configuration.rb +92 -45
  318. data/lib/avo/{cover_photo.rb → cover.rb} +2 -2
  319. data/lib/avo/current.rb +0 -11
  320. data/lib/avo/discreet_information.rb +52 -29
  321. data/lib/avo/dsl/field_parser.rb +1 -1
  322. data/lib/avo/engine.rb +41 -8
  323. data/lib/avo/error_manager.rb +1 -1
  324. data/lib/avo/fields/avatar_field.rb +19 -0
  325. data/lib/avo/fields/badge_field.rb +23 -3
  326. data/lib/avo/fields/base_field.rb +46 -1
  327. data/lib/avo/fields/concerns/dom_id.rb +17 -0
  328. data/lib/avo/fields/concerns/file_authorization.rb +4 -0
  329. data/lib/avo/fields/concerns/has_html_attributes.rb +1 -1
  330. data/lib/avo/fields/concerns/is_searchable.rb +2 -5
  331. data/lib/avo/fields/concerns/nested.rb +1 -1
  332. data/lib/avo/fields/files_field.rb +2 -2
  333. data/lib/avo/fields/frame_base_field.rb +2 -2
  334. data/lib/avo/fields/id_field.rb +5 -1
  335. data/lib/avo/fields/progress_bar_field.rb +1 -1
  336. data/lib/avo/fields/text_field.rb +1 -1
  337. data/lib/avo/photo_object.rb +12 -1
  338. data/lib/avo/plugin_manager.rb +4 -0
  339. data/lib/avo/resources/base.rb +33 -21
  340. data/lib/avo/resources/controls/actions_list.rb +3 -3
  341. data/lib/avo/resources/controls/base_control.rb +1 -1
  342. data/lib/avo/resources/items/card.rb +16 -0
  343. data/lib/avo/resources/items/header.rb +11 -0
  344. data/lib/avo/resources/items/holder.rb +18 -23
  345. data/lib/avo/resources/items/item_group.rb +5 -7
  346. data/lib/avo/resources/items/sidebar.rb +5 -9
  347. data/lib/avo/resources/items/tab.rb +8 -8
  348. data/lib/avo/resources/items/tab_group.rb +8 -10
  349. data/lib/avo/resources/resource_manager.rb +2 -1
  350. data/lib/avo/services/hq_reporter.rb +102 -0
  351. data/lib/avo/services/telemetry_service.rb +0 -1
  352. data/lib/avo/test_helpers.rb +36 -22
  353. data/lib/avo/u_i_instance.rb +60 -0
  354. data/lib/avo/version.rb +1 -1
  355. data/lib/avo/view_inquirer.rb +6 -1
  356. data/lib/avo/view_types/view_type_manager.rb +70 -0
  357. data/lib/avo.rb +30 -45
  358. data/lib/generators/avo/action_generator.rb +1 -1
  359. data/lib/generators/avo/resource_generator.rb +43 -0
  360. data/lib/generators/avo/resource_tool_generator.rb +1 -1
  361. data/lib/generators/avo/tailwindcss/install_generator.rb +0 -6
  362. data/lib/generators/avo/templates/cards/partial_card_partial.tt +1 -1
  363. data/lib/generators/avo/templates/initializer/avo.tt +7 -9
  364. data/lib/generators/avo/templates/locales/avo.ar.yml +25 -0
  365. data/lib/generators/avo/templates/locales/avo.de.yml +25 -0
  366. data/lib/generators/avo/templates/locales/avo.en.yml +25 -0
  367. data/lib/generators/avo/templates/locales/avo.es.yml +25 -0
  368. data/lib/generators/avo/templates/locales/avo.fr.yml +25 -0
  369. data/lib/generators/avo/templates/locales/avo.it.yml +25 -0
  370. data/lib/generators/avo/templates/locales/avo.ja.yml +25 -0
  371. data/lib/generators/avo/templates/locales/avo.nb.yml +25 -0
  372. data/lib/generators/avo/templates/locales/avo.nl.yml +25 -0
  373. data/lib/generators/avo/templates/locales/avo.nn.yml +25 -0
  374. data/lib/generators/avo/templates/locales/avo.pl.yml +25 -0
  375. data/lib/generators/avo/templates/locales/avo.pt-BR.yml +25 -0
  376. data/lib/generators/avo/templates/locales/avo.pt.yml +25 -0
  377. data/lib/generators/avo/templates/locales/avo.ro.yml +25 -0
  378. data/lib/generators/avo/templates/locales/avo.ru.yml +25 -0
  379. data/lib/generators/avo/templates/locales/avo.tr.yml +25 -0
  380. data/lib/generators/avo/templates/locales/{avo.uk.yml → avo.ua.yml} +26 -1
  381. data/lib/generators/avo/templates/locales/avo.zh-TW.yml +25 -0
  382. data/lib/generators/avo/templates/locales/avo.zh.yml +25 -0
  383. data/lib/generators/avo/templates/resource/resource.tt +10 -1
  384. data/lib/generators/avo/templates/resource_tools/partial.tt +20 -22
  385. data/lib/generators/avo/templates/tailwindcss/avo.tailwind.css +1 -3
  386. data/lib/generators/avo/templates/tailwindcss/tailwind.config.js +0 -7
  387. data/lib/generators/avo/templates/tool/view.tt +14 -16
  388. data/lib/generators/avo/tool_generator.rb +3 -3
  389. data/lib/generators/avo/version_generator.rb +1 -1
  390. data/lib/tasks/avo_tasks.rake +29 -25
  391. metadata +145 -97
  392. data/app/assets/stylesheets/avo.base.css +0 -130
  393. data/app/assets/stylesheets/css/breadcrumbs.css +0 -16
  394. data/app/assets/stylesheets/css/buttons.css +0 -19
  395. data/app/assets/stylesheets/css/tailwindcss/base.css +0 -1
  396. data/app/assets/stylesheets/css/tailwindcss/components.css +0 -1
  397. data/app/assets/stylesheets/css/tailwindcss/utilities.css +0 -1
  398. data/app/assets/svgs/avo/arrow-down.svg +0 -3
  399. data/app/assets/svgs/avo/arrow-up.svg +0 -3
  400. data/app/assets/svgs/avo/detach.svg +0 -8
  401. data/app/assets/svgs/avo/download-solid-reversed.svg +0 -3
  402. data/app/assets/svgs/avo/download-solid.svg +0 -3
  403. data/app/assets/svgs/avo/edit.svg +0 -5
  404. data/app/assets/svgs/avo/editor-bold.svg +0 -1
  405. data/app/assets/svgs/avo/editor-italic.svg +0 -1
  406. data/app/assets/svgs/avo/editor-link.svg +0 -1
  407. data/app/assets/svgs/avo/editor-list.svg +0 -1
  408. data/app/assets/svgs/avo/editor-strike.svg +0 -1
  409. data/app/assets/svgs/avo/editor-underline.svg +0 -1
  410. data/app/assets/svgs/avo/eye.svg +0 -1
  411. data/app/assets/svgs/avo/trash-sm.svg +0 -3
  412. data/app/assets/svgs/avo/trash.svg +0 -7
  413. data/app/components/avo/cover_photo_component.html.erb +0 -3
  414. data/app/components/avo/cover_photo_component.rb +0 -19
  415. data/app/components/avo/fields/common/badge_viewer_component.html.erb +0 -1
  416. data/app/components/avo/fields/common/badge_viewer_component.rb +0 -33
  417. data/app/components/avo/index/resource_grid_component.html.erb +0 -23
  418. data/app/components/avo/index/resource_grid_component.rb +0 -10
  419. data/app/components/avo/index/resource_map_component.html.erb +0 -16
  420. data/app/components/avo/index/resource_map_component.rb +0 -114
  421. data/app/components/avo/index/resource_table_component.html.erb +0 -52
  422. data/app/components/avo/panel_component.html.erb +0 -63
  423. data/app/components/avo/panel_component.rb +0 -43
  424. data/app/components/avo/panel_header_component.html.erb +0 -42
  425. data/app/components/avo/panel_header_component.rb +0 -31
  426. data/app/components/avo/profile_photo_component.html.erb +0 -6
  427. data/app/components/avo/profile_photo_component.rb +0 -9
  428. data/app/components/avo/sidebar/heading_component.html.erb +0 -21
  429. data/app/components/avo/sidebar/heading_component.rb +0 -9
  430. data/app/components/avo/tab_switcher_component.html.erb +0 -20
  431. data/app/components/avo/tab_switcher_component.rb +0 -45
  432. data/app/views/avo/base/close_modal_and_reload_field.turbo_stream.erb +0 -8
  433. data/app/views/avo/debug/report.html.erb +0 -37
  434. data/app/views/avo/partials/_panel_breadcrumbs.html.erb +0 -3
  435. data/app/views/avo/partials/_resource_search.html.erb +0 -16
  436. data/lib/avo/licensing/community_license.rb +0 -6
  437. data/lib/avo/licensing/h_q.rb +0 -202
  438. data/lib/avo/licensing/license.rb +0 -76
  439. data/lib/avo/licensing/license_manager.rb +0 -24
  440. data/lib/avo/licensing/nil_license.rb +0 -14
  441. data/lib/avo/licensing/pro_license.rb +0 -20
  442. data/lib/avo/licensing/request.rb +0 -20
  443. data/lib/avo/profile_photo.rb +0 -7
  444. data/lib/avo/services/debug_service.rb +0 -107
  445. data/public/avo-assets/placeholder.svg +0 -1
  446. data/tailwind.preset.js +0 -171
  447. /data/{public/avo-assets → app/assets/images/avo}/favicon.ico +0 -0
  448. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.eot +0 -0
  449. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.svg +0 -0
  450. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.ttf +0 -0
  451. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.woff +0 -0
  452. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.woff2 +0 -0
  453. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.eot +0 -0
  454. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.svg +0 -0
  455. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.ttf +0 -0
  456. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.woff +0 -0
  457. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.woff2 +0 -0
  458. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.eot +0 -0
  459. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.svg +0 -0
  460. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.ttf +0 -0
  461. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.woff +0 -0
  462. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.woff2 +0 -0
  463. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.eot +0 -0
  464. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.svg +0 -0
  465. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.ttf +0 -0
  466. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.woff +0 -0
  467. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.woff2 +0 -0
  468. /data/{public/avo-assets → app/assets/images/avo}/logo-on-white.png +0 -0
  469. /data/{public/avo-assets → app/assets/images/avo}/logo.png +0 -0
  470. /data/{public/avo-assets → app/assets/images/avo}/logomark.png +0 -0
@@ -6,24 +6,32 @@
6
6
  <% end %>
7
7
 
8
8
  <% if can_upload_file? %>
9
+ <% upload_input_id = "#{@form.object_name}_#{@field.id}" %>
9
10
  <div data-controller="clear-input">
10
- <div class="mt-2 flex items-center">
11
+ <%= render Avo::UI::FileUploadInputComponent.new(
12
+ id: upload_input_id,
13
+ title: @field.is_audio ? "Upload audio" : "Click to upload",
14
+ subtitle: "or drag and drop",
15
+ description: @field.accept.present? ? "Accepts #{@field.accept}" : nil,
16
+ disabled: disabled?,
17
+ ) do %>
11
18
  <%= @form.file_field @field.id,
19
+ id: upload_input_id,
12
20
  accept: @field.accept,
13
21
  data: @field.get_html(:data, view: view, element: :input)
14
22
  .merge(
15
23
  action: "change->clear-input#showClearButton",
16
- clear_input_target: "input"
24
+ clear_input_target: "input",
25
+ file_upload_input_target: "input"
17
26
  ),
18
27
  direct_upload: @field.direct_upload,
19
28
  disabled: disabled?,
20
29
  style: @field.get_html(:style, view: view, element: :input),
21
- autofocus: @autofocus,
22
- class: "w-full"
30
+ autofocus: @autofocus
23
31
  %>
24
- </div>
32
+ <% end %>
25
33
  <%= content_tag :button,
26
- class: "self-center hidden font-semibold text-xs text-red-600 p-1",
34
+ class: "self-center hidden font-semibold text-xs text-red-600 p-1 cursor-pointer mt-2",
27
35
  id: :reset,
28
36
  type: :button,
29
37
  data: {
@@ -2,14 +2,21 @@
2
2
  <%= render Avo::Fields::Common::Files::ListViewerComponent.new(field: @field, resource: @resource) if @field.value.present? %>
3
3
 
4
4
  <% if can_upload_file? %>
5
+ <% upload_input_id = "#{@form.object_name}_#{@field.id}" %>
5
6
  <div data-controller="clear-input">
6
- <div class="mt-2 flex items-center">
7
+ <%= render Avo::UI::FileUploadInputComponent.new(
8
+ id: upload_input_id,
9
+ description: @field.accept.present? ? "Accepts #{@field.accept}" : "Accepts .jpg and .png",
10
+ disabled: disabled?,
11
+ ) do %>
7
12
  <%= @form.file_field @field.id,
13
+ id: upload_input_id,
8
14
  accept: @field.accept,
9
15
  data: @field.get_html(:data, view: view, element: :input)
10
16
  .merge(
11
17
  action: "change->clear-input#showClearButton",
12
- clear_input_target: "input"
18
+ clear_input_target: "input",
19
+ file_upload_input_target: "input"
13
20
  ),
14
21
  direct_upload: @field.direct_upload,
15
22
  disabled: disabled?,
@@ -17,9 +24,9 @@
17
24
  style: @field.get_html(:style, view: view, element: :input),
18
25
  autofocus: @autofocus
19
26
  %>
20
- </div>
27
+ <% end %>
21
28
  <%= content_tag :button,
22
- class: "self-center hidden font-semibold text-xs text-red-600 p-1",
29
+ class: "self-center hidden font-semibold text-xs text-red-600 p-1 cursor-pointer",
23
30
  id: :reset,
24
31
  type: :button,
25
32
  data: {
@@ -1,13 +1,13 @@
1
1
  <% if @field.value %>
2
- <turbo-frame id="<%= @field.turbo_frame %>" src="<%= @field.frame_url %>" loading=<%= turbo_frame_loading %> target="_top" class="block">
2
+ <%= turbo_frame_tag @field.turbo_frame, src: @field.frame_url, loading: turbo_frame_loading, target: "_top", class: "block" do %>
3
3
  <%= render(Avo::LoadingComponent.new(title: @field.name)) %>
4
- </turbo-frame>
4
+ <% end %>
5
5
  <% else %>
6
- <%= render Avo::PanelComponent.new(name: @field.name) do |c| %>
7
- <% c.with_tools do %>
6
+ <%= render ui.panel(title: @field.name) do |panel| %>
7
+ <% panel.with_controls do %>
8
8
  <% if can_attach? %>
9
9
  <%= a_link attach_path,
10
- icon: 'heroicons/outline/link',
10
+ icon: 'tabler/outline/link',
11
11
  color: :primary,
12
12
  style: :text,
13
13
  data: {
@@ -17,20 +17,21 @@
17
17
  <%= t('avo.attach_item', item: @field.name.humanize(capitalize: false)) %>
18
18
  <% end %>
19
19
  <% end %>
20
+
20
21
  <% if can_see_the_create_button? %>
21
22
  <%= a_link create_path,
22
- icon: 'heroicons/outline/plus',
23
+ icon: 'tabler/outline/plus',
23
24
  'data-target': 'create',
24
25
  'data-turbo-frame': '_top',
25
26
  style: :primary,
26
- color: :primary do %>
27
+ color: :accent do %>
27
28
  <%= t('avo.create_new_item', item: @field.name.humanize(capitalize: false) ) %>
28
29
  <% end %>
29
30
  <% end %>
30
31
  <% end %>
31
32
 
32
- <% c.with_body do %>
33
- <div class="py-8 flex justify-center items-center">
33
+ <% panel.with_body do %>
34
+ <div class="flex items-center justify-center py-8">
34
35
  <%= empty_state by_association: params[:related_name].present? %>
35
36
  </div>
36
37
  <% end %>
@@ -1,3 +1,3 @@
1
- <div <% if @index == 0 %> class="overflow-hidden rounded-t" <% end %> data-field-id="<%= @field.id %>">
1
+ <div class="<%= class_names("w-full",{'overflow-hidden rounded-t': @index == 0}) %>" data-field-id="<%= @field.id %>">
2
2
  <%= render Avo::Fields::Common::HeadingComponent.new field: @field %>
3
3
  </div>
@@ -1,3 +1,3 @@
1
- <div <% if @index == 0 %> class="overflow-hidden rounded-t" <% end %> data-field-id="<%= @field.id %>">
1
+ <%= tag.div class: class_names(@field.width_class, "overflow-hidden rounded-t": @index == 0), data: {field_id: @field.id} do %>
2
2
  <%= render Avo::Fields::Common::HeadingComponent.new field: @field %>
3
- </div>
3
+ <% end %>
@@ -1,7 +1,8 @@
1
1
  <%= field_wrapper(**field_wrapper_args) do %>
2
- <div class="relative" data-controller="<%= @field.revealable ? 'password-visibility' : nil %>">
2
+ <div class="field-wrapper__input" data-controller="<%= @field.revealable ? 'password-visibility' : nil %>">
3
3
  <%= @form.password_field @field.id,
4
4
  value: @field.value,
5
+ type: "password",
5
6
  class: classes("w-full"),
6
7
  data: @field.get_html(:data, view: view, element: :input).merge(password_visibility_target: "input"),
7
8
  disabled: disabled?,
@@ -12,12 +13,9 @@
12
13
  %>
13
14
 
14
15
  <% if @field.revealable %>
15
- <button data-action="password-visibility#toggle" type="button" class="text-gray-500 absolute inset-y-0 right-0 pr-3 flex items-center cursor-pointer">
16
- <!-- Heroicon name: outline/eye -->
17
- <%= helpers.svg('heroicons/outline/eye', class: "size-6", data: { 'password-visibility-target': 'icon' }) %>
18
-
19
- <!-- Heroicon name: outline/eye-off -->
20
- <%= helpers.svg('heroicons/outline/eye-off', class: "size-6 hidden", data: { 'password-visibility-target': 'icon' }) %>
16
+ <button data-action="password-visibility#toggle" type="button" class="input-field__password-toggle">
17
+ <%= helpers.svg('tabler/outline/eye', data: { 'password-visibility-target': 'icon' }) %>
18
+ <%= helpers.svg('tabler/outline/eye-off', class: "hidden", data: { 'password-visibility-target': 'icon' }) %>
21
19
  </button>
22
20
  <% end %>
23
21
  </div>
@@ -1,7 +1,7 @@
1
1
  <%= field_wrapper(**field_wrapper_args) do %>
2
2
  <%= content_tag :div, data: {controller: "progress-bar-field"} do %>
3
3
  <% if @field.display_value %>
4
- <div class="text-center text-sm font-semibold w-full leading-none mb-1">
4
+ <div class="text-center text-sm font-semibold w-full leading-none mb-1 mt-1">
5
5
  <span data-progress-bar-field-target="label"><%= @field.value %></span><%= @field.value_suffix if @field.value_suffix.present? %>
6
6
  </div>
7
7
  <% end %>
@@ -1,7 +1,7 @@
1
1
  <%= field_wrapper(**field_wrapper_args) do %>
2
2
  <div class="flex flex-col gap-2">
3
3
  <% @field.options.each do |key, value| %>
4
- <div>
4
+ <div class="flex items-center gap-1">
5
5
  <%= form.radio_button @field.id, key, checked: (@field.value.to_s == key.to_s) %>
6
6
  <%= form.label @field.id, value, value: key %>
7
7
  </div>
@@ -3,24 +3,22 @@
3
3
  class Avo::Fields::ShowComponent < Avo::BaseComponent
4
4
  include Avo::ResourcesHelper
5
5
 
6
- attr_reader :compact
7
6
  attr_reader :field
8
7
  attr_reader :index
9
8
  attr_reader :kwargs
10
9
  attr_reader :resource
11
10
  attr_reader :stacked
12
- attr_reader :short
13
11
  attr_reader :view
12
+ attr_reader :full_width
14
13
 
15
- def initialize(field: nil, resource: nil, index: 0, form: nil, compact: false, short: false, stacked: nil, **kwargs)
16
- @compact = compact
14
+ def initialize(field: nil, resource: nil, index: 0, form: nil, stacked: nil, full_width: nil, **kwargs)
17
15
  @field = field
18
16
  @index = index
19
17
  @resource = resource
20
18
  @stacked = stacked
21
- @short = short
22
19
  @kwargs = kwargs
23
20
  @view = Avo::ViewInquirer.new("show")
21
+ @full_width = full_width
24
22
  end
25
23
 
26
24
  def wrapper_data
@@ -48,12 +46,11 @@ class Avo::Fields::ShowComponent < Avo::BaseComponent
48
46
 
49
47
  def field_wrapper_args
50
48
  {
51
- compact: compact,
52
49
  field: field,
53
50
  index: index,
54
51
  resource: resource,
55
- short: short,
56
52
  stacked: stacked,
53
+ full_width: full_width,
57
54
  view: view
58
55
  }
59
56
  end
@@ -3,15 +3,14 @@
3
3
  <%= @form.hidden_field @field.id,
4
4
  value: @field.value,
5
5
  data: { stars_field_target: "valueInput" } %>
6
-
7
- <div class="rating-group flex items-center gap-1 mb-3"
6
+ <div class="rating-group flex items-center gap-1"
8
7
  data-component="avo/fields/common/stars_component"
9
8
  data-stars-field-target="starsContainer"
10
9
  data-max="<%= @field.max %>"
11
10
  data-current-value="<%= @field.value || 0 %>">
12
11
  <% (1..@field.max).each do |rating_value| %>
13
- <%= helpers.svg('heroicons/solid/star',
14
- class: class_names("star inline-block mr-0.5", filled: rating_value <= (@value || 0)),
12
+ <%= helpers.svg('heroicons/solid/star',
13
+ class: class_names("star inline-block me-0.5", filled: rating_value <= (@value || 0)),
15
14
  data: {
16
15
  star_value: rating_value,
17
16
  stars_field_target: "star",
@@ -23,7 +23,7 @@
23
23
  %>
24
24
  <%# real field %>
25
25
  <%= @form.text_field @field.id,
26
- class: classes("hidden w-full !py-0 min-h-10 items-center border-primary-500 focus-within:border-primary-500"),
26
+ class: classes("!hidden w-full"),
27
27
  data: {
28
28
  tags_field_target: :input,
29
29
  },
@@ -1,9 +1,8 @@
1
- <div class="flex px-2 py-1 rounded bg-gray-100 text-sm text-gray-800 font-normal flex-shrink-0"
2
- <% if @title.present? %>
3
- data-tippy="tooltip"
4
- title="<%= @title %>"
5
- <% end %>
6
- data-target="tag-component"
7
- >
1
+ <%= tag.div class: "flex px-2 py-0.5 rounded-sm bg-gray-100 text-sm text-gray-800 font-normal shrink-0",
2
+ data: {
3
+ target: "tag-component",
4
+ tippy: (@title.present? ? "tooltip" : nil),
5
+ }.compact,
6
+ title: @title do %>
8
7
  <%= @label %>
9
- </div>
8
+ <% end %>
@@ -14,7 +14,7 @@
14
14
  } do %>
15
15
  <%= @form.text_field @field.id,
16
16
  value: @field.edit_formatted_value,
17
- class: classes("w-full #{"hidden" unless params[:avo_show_hidden_inputs]}"),
17
+ class: classes("w-full #{"!hidden" unless params[:avo_show_hidden_inputs]}"),
18
18
  data: {
19
19
  date_field_target: :input,
20
20
  placeholder: @field.placeholder,
@@ -37,7 +37,7 @@
37
37
  style: @field.get_html(:style, view: view, element: :input)
38
38
  %>
39
39
  <%= content_tag :button,
40
- class: "absolute right-0 self-center mr-4 uppercase font-semibold text-xs disabled:cursor-not-allowed disabled:opacity-50",
40
+ class: "absolute end-0 self-center me-2 uppercase font-semibold text-xs disabled:cursor-not-allowed disabled:opacity-50 text-content-secondary cursor-pointer",
41
41
  id: :reset,
42
42
  type: :button,
43
43
  title: t("avo.reset").capitalize,
@@ -46,7 +46,7 @@
46
46
  action: "click->date-field#clear",
47
47
  tippy: :tooltip
48
48
  } do %>
49
- <%= helpers.svg "avo/times", class: "h-4" %>
49
+ <%= helpers.svg "tabler/outline/x", class: "h-4" %>
50
50
  <% end %>
51
51
  <% end %>
52
52
  <% end %>
@@ -1,4 +1,4 @@
1
- <%= field_wrapper(**field_wrapper_args) do %>
1
+ <%= field_wrapper(**field_wrapper_args, full_width: true) do %>
2
2
  <%= content_tag :div,
3
3
  class: "tiptap__field",
4
4
  data: {
@@ -20,45 +20,45 @@
20
20
  <%= content_tag :div, class: "tiptap__toolbar-area" do %>
21
21
  <%= content_tag :div, class: "tiptap__button-group" do %>
22
22
  <%= button_tag type: "button", class: "tiptap__button tiptap__button--bold", data: { action: "click->tiptap-field#bold" } do %>
23
- <%= helpers.svg("avo/editor-bold") %>
23
+ <%= helpers.svg("tabler/outline/bold", class: "size-5") %>
24
24
  <% end %>
25
25
 
26
26
  <%= button_tag type: "button", class: "tiptap__button tiptap__button--italic", data: { action: "click->tiptap-field#italic" } do %>
27
- <%= helpers.svg("avo/editor-italic") %>
27
+ <%= helpers.svg("tabler/outline/italic", class: "size-5") %>
28
28
  <% end %>
29
29
 
30
30
  <%= button_tag type: "button", class: "tiptap__button tiptap__button--underline", data: { action: "click->tiptap-field#underline" } do %>
31
- <%= helpers.svg("avo/editor-underline") %>
31
+ <%= helpers.svg("tabler/outline/underline", class: "size-5") %>
32
32
  <% end %>
33
33
 
34
34
  <%= button_tag type: "button", class: "tiptap__button tiptap__button--strike", data: { action: "click->tiptap-field#strike" } do %>
35
- <%= helpers.svg("avo/editor-strike") %>
35
+ <%= helpers.svg("tabler/outline/strikethrough", class: "size-5") %>
36
36
  <% end %>
37
37
  <% end %>
38
38
 
39
39
  <%= content_tag :div, class: "tiptap__button-group" do %>
40
40
  <%= button_tag type: "button", class: "tiptap__button tiptap__button--ul", data: { action: "click->tiptap-field#unorderedList" } do %>
41
- <%= helpers.svg("avo/editor-list") %>
41
+ <%= helpers.svg("tabler/outline/list", class: "size-5") %>
42
42
  <% end %>
43
43
 
44
44
  <%= button_tag type: "button", class: "tiptap__button tiptap__button--ol", data: { action: "click->tiptap-field#orderedList" } do %>
45
- <%= helpers.svg("avo/editor-ordered-list") %>
45
+ <%= helpers.svg("tabler/outline/list-numbers", class: "size-5") %>
46
46
  <% end %>
47
47
  <% end %>
48
48
 
49
49
  <%= content_tag :div, class: "tiptap__button-group" do %>
50
50
  <%= button_tag type: "button", disabled: true, class: "tiptap__button tiptap__button--link", data: { action: "click->tiptap-field#toggleLinkArea" } do %>
51
- <%= helpers.svg("avo/editor-link") %>
51
+ <%= helpers.svg("tabler/outline/link", class: "size-5") %>
52
52
  <% end %>
53
53
  <% end %>
54
54
 
55
55
  <%= content_tag :div, class: "tiptap__button-group tiptap__link-area hidden" do %>
56
56
  <%= text_field_tag nil, nil, class: "tiptap__link-field", placeholder: "Enter URL here ...", data: { action: "keydown->tiptap-field#preventEnter" } %>
57
57
  <%= button_tag type: "button", class: "tiptap__button", data: { action: "click->tiptap-field#setLink" } do %>
58
- <%= helpers.svg("heroicons/outline/check-circle") %>
58
+ <%= helpers.svg("tabler/outline/circle-check", class: "size-5") %>
59
59
  <% end %>
60
60
  <%= button_tag type: "button", class: "tiptap__button tiptap__link-button--unset", data: { action: "click->tiptap-field#unsetLink" } do %>
61
- <%= helpers.svg("avo/trash") %>
61
+ <%= helpers.svg("tabler/outline/trash", class: "size-5") %>
62
62
  <% end %>
63
63
  <% end %>
64
64
  <% end %>
@@ -67,7 +67,7 @@
67
67
 
68
68
  <%= @form.text_area @field.id,
69
69
  value: @field.value,
70
- class: classes("w-full hidden"),
70
+ class: classes("w-full !hidden"),
71
71
  data: {
72
72
  tiptap_field_target: "input",
73
73
  **@field.get_html(:data, view: view, element: :input)
@@ -1,10 +1,10 @@
1
1
  <%= field_wrapper(**field_wrapper_args, data: { controller: 'hidden-input' }) do %>
2
- <div data-controller="hidden-input" class="flex w-full">
3
- <% unless @field.always_show %>
4
- <%= link_to t('avo.show_content'), 'javascript:void(0);', class: 'font-bold inline-block', data: { action: 'click->hidden-input#showContent' } %>
5
- <% end %>
2
+ <div data-controller="hidden-input" class="flex flex-col w-full">
6
3
  <div class="<%= class_names("tiptap__content py-2 max-w-4xl", "hidden": !@field.always_show) %> " data-hidden-input-target="content">
7
4
  <%= sanitize @field.value.to_s %>
8
5
  </div>
6
+ <% unless @field.always_show %>
7
+ <%= link_to t('avo.show_content'), 'javascript:void(0);', class: 'font-bold inline-block', data: { action: 'click->hidden-input#showContent' } %>
8
+ <% end %>
9
9
  </div>
10
10
  <% end %>
@@ -1,4 +1,4 @@
1
- <%= field_wrapper(**field_wrapper_args) do %>
1
+ <%= field_wrapper(**field_wrapper_args, full_width: true) do %>
2
2
  <%= content_tag :div,
3
3
  class: class_names("relative block overflow-x-auto max-w-4xl", @input_id),
4
4
  data: do %>
@@ -14,7 +14,7 @@
14
14
  <% end %>
15
15
  <%= @form.text_area @field.id,
16
16
  value: @field.value.try(:to_trix_html) || @field.value,
17
- class: classes("w-full hidden"),
17
+ class: classes("w-full !hidden"),
18
18
  data: @field.get_html(:data, view: view, element: :input),
19
19
  disabled: disabled?,
20
20
  id: @input_id,
@@ -1,8 +1,7 @@
1
1
  <div data-controller="toggle" data-toggle-exemption-containers-value='[".flatpickr-calendar"]' data-component-name="<%= self.class.to_s.underscore %>">
2
2
  <div class="relative w-full flex justify-between">
3
- <%= a_button class: 'focus:outline-none',
4
- color: :primary,
5
- size: :sm,
3
+ <%= a_button class: 'focus:outline-hidden',
4
+ style: :primary,
6
5
  icon: "avo/filter",
7
6
  title: t('avo.click_to_reveal_filters'),
8
7
  'data-button': 'resource-filters',
@@ -10,23 +9,16 @@
10
9
  'data-tippy': 'tooltip' do %>
11
10
  <%= t 'avo.filters' %>
12
11
  <% if @applied_filters.present? %>
13
- <span class="ml-1">(<%= "#{@applied_filters.count} #{t('avo.applied', count: @applied_filters.count)}" %>)</span>
12
+ <span class="ms-1">(<%= "#{@applied_filters.count} #{t('avo.applied', count: @applied_filters.count)}" %>)</span>
14
13
  <% end %>
15
14
  <% end %>
16
- <div
17
- class="absolute block inset-auto sm:right-0 top-full bg-white min-w-[300px] mt-2 z-40 shadow-modal rounded divide-y divide-gray-300 <%= 'hidden' unless params[:keep_filters_panel_open] == '1' %>"
18
- data-toggle-target="panel"
19
- data-transition-enter="transition ease-out duration-100"
20
- data-transition-enter-start="transform opacity-0 -translate-y-1"
21
- data-transition-enter-end="transform opacity-100 translate-y-0"
22
- data-transition-leave="transition ease-in duration-75"
23
- data-transition-leave-start="transform opacity-100 translate-y-0"
24
- data-transition-leave-end="transform opacity-0 -translate-y-1"
25
- >
15
+ <%= tag.div class: 'absolute block inset-auto sm:end-0 top-full bg-white min-w-[300px] mt-2 z-40 shadow-modal rounded-sm divide-y divide-gray-300 css-animate-slide-down',
16
+ data: {toggle_target: 'panel'},
17
+ hidden: params[:keep_filters_panel_open] != '1' do %>
26
18
  <% @filters.each do |filter| %>
27
19
  <%= render partial: filter.class.template, locals: {filter: filter} %>
28
20
  <% end %>
29
- <div class="p-4 border-gray-300 border-t">
21
+ <div class="p-4 border-tertiary border-t">
30
22
  <% if @applied_filters.present? %>
31
23
  <%= a_link reset_path, data: {turbo_frame: params[:turbo_frame]}, color: :gray, size: :sm, class: 'w-full justify-center' do %>
32
24
  <%= t('avo.reset_filters') %>
@@ -37,6 +29,6 @@
37
29
  <% end %>
38
30
  <% end %>
39
31
  </div>
40
- </div>
32
+ <% end %>
41
33
  </div>
42
34
  </div>
@@ -1,11 +1,18 @@
1
1
  <%= content_tag :td,
2
- class: "group/clipboard min-h-[3rem] px-3 leading-tight whitespace-nowrap h-full text-slate-800 text-base #{classes}",
2
+ class: class_names(
3
+ "group/clipboard px-2 leading-none whitespace-nowrap h-full text-content text-sm",
4
+ @classes,
5
+ @field.get_html(:classes, view: @view, element: :wrapper),
6
+ {
7
+ "py-4": !@flush
8
+ }
9
+ ),
3
10
  data: {
4
11
  field_id: @field.id,
5
12
  field_type: @field.type,
6
13
  **stimulus_attributes
7
14
  },
8
- style: style do %>
15
+ style: @field.get_html(:style, view: @view, element: :wrapper) do %>
9
16
  <% if render_dash? %>
10
17
 
11
18
  <% else %>
@@ -13,22 +13,6 @@ class Avo::Index::FieldWrapperComponent < Avo::BaseComponent
13
13
  @classes = @args.dig(:class) || ""
14
14
  end
15
15
 
16
- def classes
17
- result = @classes
18
-
19
- unless @flush
20
- result += " py-3"
21
- end
22
-
23
- result += " #{@field.get_html(:classes, view: @view, element: :wrapper)}"
24
-
25
- result
26
- end
27
-
28
- def style
29
- @field.get_html(:style, view: @view, element: :wrapper)
30
- end
31
-
32
16
  def stimulus_attributes
33
17
  attributes = {}
34
18
 
@@ -1,3 +1,17 @@
1
- <div class="absolute bg-gray-50 w-full h-full">
2
- <%= image_tag Avo.configuration.branding.placeholder, class: "relative transform -translate-x-1/2 -translate-y-1/2 h-20 text-gray-400 inset-auto top-1/2 left-1/2", loading: :lazy %>
1
+ <div
2
+ class="absolute bg-secondary dark:bg-tertiary size-full content-center"
3
+ data-controller="grid-cover-empty-state"
4
+ data-action="mousemove->grid-cover-empty-state#mousemove mouseleave->grid-cover-empty-state#mouseleave"
5
+ >
6
+ <%# Decorative icons — purely visual, hidden from screen readers %>
7
+ <% icon_items.each do |icon| %>
8
+ <span
9
+ aria-hidden="true"
10
+ class="grid-card__icon grid-card__icon--float"
11
+ data-grid-cover-empty-state-target="icon"
12
+ style="<%= icon[:v_edge] %>: <%= icon[:v_px] %>px; <%= icon[:h_edge] %>: <%= icon[:h_px] %>px; rotate: <%= icon[:rotation] %>deg; animation-duration: <%= icon[:duration] %>s; animation-delay: <%= icon[:delay] %>s; color: color-mix(in srgb, var(--color-content) <%= icon[:intensity] %>%, transparent);"
13
+ ><%= helpers.svg icon[:path], class: "size-4" %></span>
14
+ <% end %>
15
+
16
+ <span data-grid-cover-empty-state-target="icon" data-repulsion-scale="0.125" class="grid-card__avocado"><%= svg Avo.configuration.branding.placeholder, class: "mx-auto h-20 text-content-secondary" %></span>
3
17
  </div>
@@ -1,4 +1,94 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  class Avo::Index::GridCoverEmptyStateComponent < Avo::BaseComponent
4
+ ICONS = %w[
5
+ tabler/outline/star
6
+ tabler/outline/bolt
7
+ tabler/outline/clock
8
+ tabler/outline/heart
9
+ tabler/outline/leaf
10
+ tabler/outline/sparkles
11
+ tabler/outline/rocket
12
+ tabler/outline/diamond
13
+ ].freeze
14
+
15
+ # Per-icon animation timing and color intensity (stable across all layouts)
16
+ ICON_META = [
17
+ {duration: 3.8, delay: 0.0, intensity: 15},
18
+ {duration: 4.2, delay: 0.4, intensity: 18},
19
+ {duration: 3.5, delay: 0.8, intensity: 13},
20
+ {duration: 4.6, delay: 1.2, intensity: 20},
21
+ {duration: 3.2, delay: 1.6, intensity: 16},
22
+ {duration: 4.8, delay: 2.0, intensity: 14},
23
+ {duration: 3.6, delay: 2.4, intensity: 19},
24
+ {duration: 4.4, delay: 2.8, intensity: 17}
25
+ ].freeze
26
+
27
+ # 5 preset spatial layouts — each is an array of 8 position+rotation hashes.
28
+ # v_edge: "top"|"bottom", h_edge: CSS logical "inset-inline-start"|"inset-inline-end"
29
+ # v_px / h_px: distance in px from that edge. rotation: degrees.
30
+ LAYOUTS = [
31
+ # Layout 1 — scattered, original feel
32
+ [
33
+ {v_edge: "top", v_px: 28, h_edge: "inset-inline-start", h_px: 28, rotation: 12},
34
+ {v_edge: "top", v_px: 32, h_edge: "inset-inline-end", h_px: 32, rotation: -12},
35
+ {v_edge: "top", v_px: 72, h_edge: "inset-inline-start", h_px: 24, rotation: 0},
36
+ {v_edge: "bottom", v_px: 28, h_edge: "inset-inline-start", h_px: 44, rotation: 6},
37
+ {v_edge: "top", v_px: 40, h_edge: "inset-inline-start", h_px: 72, rotation: -6},
38
+ {v_edge: "bottom", v_px: 32, h_edge: "inset-inline-end", h_px: 32, rotation: 12},
39
+ {v_edge: "top", v_px: 28, h_edge: "inset-inline-end", h_px: 52, rotation: -6},
40
+ {v_edge: "bottom", v_px: 28, h_edge: "inset-inline-end", h_px: 60, rotation: 6}
41
+ ],
42
+ # Layout 2 — corners and mid-edges
43
+ [
44
+ {v_edge: "top", v_px: 24, h_edge: "inset-inline-start", h_px: 20, rotation: -8},
45
+ {v_edge: "top", v_px: 24, h_edge: "inset-inline-end", h_px: 20, rotation: 8},
46
+ {v_edge: "bottom", v_px: 24, h_edge: "inset-inline-start", h_px: 20, rotation: 8},
47
+ {v_edge: "bottom", v_px: 24, h_edge: "inset-inline-end", h_px: 20, rotation: -8},
48
+ {v_edge: "top", v_px: 24, h_edge: "inset-inline-start", h_px: 68, rotation: 0},
49
+ {v_edge: "bottom", v_px: 24, h_edge: "inset-inline-start", h_px: 68, rotation: 0},
50
+ {v_edge: "top", v_px: 60, h_edge: "inset-inline-start", h_px: 20, rotation: 12},
51
+ {v_edge: "top", v_px: 60, h_edge: "inset-inline-end", h_px: 20, rotation: -12}
52
+ ],
53
+ # Layout 3 — diagonal scatter
54
+ [
55
+ {v_edge: "top", v_px: 24, h_edge: "inset-inline-start", h_px: 36, rotation: 15},
56
+ {v_edge: "top", v_px: 48, h_edge: "inset-inline-end", h_px: 24, rotation: -10},
57
+ {v_edge: "top", v_px: 66, h_edge: "inset-inline-start", h_px: 28, rotation: 5},
58
+ {v_edge: "bottom", v_px: 66, h_edge: "inset-inline-end", h_px: 28, rotation: -5},
59
+ {v_edge: "bottom", v_px: 48, h_edge: "inset-inline-start", h_px: 24, rotation: 10},
60
+ {v_edge: "bottom", v_px: 24, h_edge: "inset-inline-end", h_px: 36, rotation: -15},
61
+ {v_edge: "top", v_px: 36, h_edge: "inset-inline-start", h_px: 72, rotation: 8},
62
+ {v_edge: "bottom", v_px: 36, h_edge: "inset-inline-end", h_px: 60, rotation: -8}
63
+ ],
64
+ # Layout 4 — even frame
65
+ [
66
+ {v_edge: "top", v_px: 24, h_edge: "inset-inline-start", h_px: 44, rotation: 12},
67
+ {v_edge: "top", v_px: 24, h_edge: "inset-inline-end", h_px: 44, rotation: -12},
68
+ {v_edge: "top", v_px: 60, h_edge: "inset-inline-start", h_px: 20, rotation: 0},
69
+ {v_edge: "top", v_px: 60, h_edge: "inset-inline-end", h_px: 20, rotation: 0},
70
+ {v_edge: "bottom", v_px: 60, h_edge: "inset-inline-start", h_px: 20, rotation: 6},
71
+ {v_edge: "bottom", v_px: 60, h_edge: "inset-inline-end", h_px: 20, rotation: -6},
72
+ {v_edge: "bottom", v_px: 24, h_edge: "inset-inline-start", h_px: 44, rotation: -12},
73
+ {v_edge: "bottom", v_px: 24, h_edge: "inset-inline-end", h_px: 44, rotation: 12}
74
+ ],
75
+ # Layout 5 — asymmetric cluster
76
+ [
77
+ {v_edge: "top", v_px: 24, h_edge: "inset-inline-start", h_px: 20, rotation: 8},
78
+ {v_edge: "top", v_px: 40, h_edge: "inset-inline-start", h_px: 52, rotation: -8},
79
+ {v_edge: "top", v_px: 24, h_edge: "inset-inline-end", h_px: 36, rotation: 12},
80
+ {v_edge: "top", v_px: 66, h_edge: "inset-inline-end", h_px: 20, rotation: -5},
81
+ {v_edge: "bottom", v_px: 24, h_edge: "inset-inline-start", h_px: 36, rotation: -12},
82
+ {v_edge: "bottom", v_px: 52, h_edge: "inset-inline-start", h_px: 20, rotation: 8},
83
+ {v_edge: "bottom", v_px: 24, h_edge: "inset-inline-end", h_px: 20, rotation: 10},
84
+ {v_edge: "bottom", v_px: 40, h_edge: "inset-inline-end", h_px: 52, rotation: -10}
85
+ ]
86
+ ].freeze
87
+
88
+ def icon_items
89
+ layout = LAYOUTS.sample
90
+ ICONS.each_with_index.map do |path, i|
91
+ ICON_META[i].merge(layout[i]).merge(path: path)
92
+ end
93
+ end
4
94
  end