avo 3.30.4 → 4.0.0.beta.1

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 (466) 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 +13942 -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/stylesheets/application.css +241 -0
  12. data/app/assets/stylesheets/css/button-group.css +23 -0
  13. data/app/assets/stylesheets/css/components/avatar.css +128 -0
  14. data/app/assets/stylesheets/css/components/breadcrumbs.css +43 -0
  15. data/app/assets/stylesheets/css/components/button.css +343 -0
  16. data/app/assets/stylesheets/css/components/color_scheme_switcher.css +226 -0
  17. data/app/assets/stylesheets/css/components/discreet_information.css +49 -0
  18. data/app/assets/stylesheets/css/components/field-wrapper.css +107 -0
  19. data/app/assets/stylesheets/css/components/grid.css +61 -0
  20. data/app/assets/stylesheets/css/components/input.css +312 -0
  21. data/app/assets/stylesheets/css/components/modal.css +228 -0
  22. data/app/assets/stylesheets/css/components/tooltip.css +25 -0
  23. data/app/assets/stylesheets/css/components/ui/badge.css +143 -0
  24. data/app/assets/stylesheets/css/components/ui/card.css +95 -0
  25. data/app/assets/stylesheets/css/components/ui/checkbox.css +50 -0
  26. data/app/assets/stylesheets/css/components/ui/description_list.css +3 -0
  27. data/app/assets/stylesheets/css/components/ui/dropdown.css +68 -0
  28. data/app/assets/stylesheets/css/components/ui/file_upload_input.css +94 -0
  29. data/app/assets/stylesheets/css/components/ui/file_upload_item.css +78 -0
  30. data/app/assets/stylesheets/css/components/ui/panel.css +59 -0
  31. data/app/assets/stylesheets/css/components/ui/panel_header.css +48 -0
  32. data/app/assets/stylesheets/css/components/ui/radio.css +22 -0
  33. data/app/assets/stylesheets/css/components/ui/tabs.css +74 -0
  34. data/app/assets/stylesheets/css/css-animations.css +54 -0
  35. data/app/assets/stylesheets/css/fields/code.css +75 -9
  36. data/app/assets/stylesheets/css/fields/easy-mde.css +7 -0
  37. data/app/assets/stylesheets/css/fields/key_value.css +97 -0
  38. data/app/assets/stylesheets/css/fields/progress.css +4 -4
  39. data/app/assets/stylesheets/css/fields/status.css +7 -1
  40. data/app/assets/stylesheets/css/fields/tags.css +33 -16
  41. data/app/assets/stylesheets/css/fields/tiptap.css +17 -15
  42. data/app/assets/stylesheets/css/fields/trix.css +62 -2
  43. data/app/assets/stylesheets/css/fonts.css +24 -24
  44. data/app/assets/stylesheets/css/layout.css +135 -0
  45. data/app/assets/stylesheets/css/pagination.css +114 -78
  46. data/app/assets/stylesheets/css/resource-controls.css +13 -0
  47. data/app/assets/stylesheets/css/search.css +22 -12
  48. data/app/assets/stylesheets/css/sidebar.css +310 -24
  49. data/app/assets/stylesheets/css/table.css +60 -0
  50. data/app/assets/stylesheets/css/tooltips.css +1 -1
  51. data/app/assets/stylesheets/css/typography.css +10 -1
  52. data/app/assets/stylesheets/css/variables.css +318 -0
  53. data/app/assets/svgs/avo/moon-plus-plus.svg +1 -0
  54. data/app/components/avo/actions_component.html.erb +30 -36
  55. data/app/components/avo/actions_component.rb +8 -11
  56. data/app/components/avo/alert_component.html.erb +3 -3
  57. data/app/components/avo/alert_component.rb +1 -1
  58. data/app/components/avo/asset_manager/javascript_component.html.erb +1 -2
  59. data/app/components/avo/backtrace_alert_component.html.erb +4 -4
  60. data/app/components/avo/base_component.rb +3 -4
  61. data/app/components/avo/breadcrumb_element_component.html.erb +17 -0
  62. data/app/components/avo/breadcrumb_element_component.rb +21 -0
  63. data/app/components/avo/breadcrumbs_component.html.erb +19 -0
  64. data/app/components/avo/breadcrumbs_component.rb +5 -0
  65. data/app/components/avo/button_component.rb +20 -126
  66. data/app/components/avo/clipboard_component.html.erb +3 -2
  67. data/app/components/avo/clipboard_component.rb +1 -1
  68. data/app/components/avo/component_missing_component.rb +11 -0
  69. data/app/components/avo/cover_component.html.erb +3 -0
  70. data/app/components/avo/cover_component.rb +25 -0
  71. data/app/components/avo/debug/status_component.html.erb +59 -0
  72. data/app/components/avo/debug/status_component.rb +30 -0
  73. data/app/components/avo/description_list_component.rb +11 -0
  74. data/app/components/avo/discreet_information_component.html.erb +31 -6
  75. data/app/components/avo/discreet_information_component.rb +23 -32
  76. data/app/components/avo/divider_component.html.erb +2 -2
  77. data/app/components/avo/empty_state_component.html.erb +2 -9
  78. data/app/components/avo/empty_state_component.rb +0 -8
  79. data/app/components/avo/field_wrapper_component.html.erb +21 -22
  80. data/app/components/avo/field_wrapper_component.rb +9 -18
  81. data/app/components/avo/fields/avatar_field/index_component.html.erb +9 -0
  82. data/app/components/avo/fields/avatar_field/index_component.rb +4 -0
  83. data/app/components/avo/fields/badge_field/index_component.html.erb +7 -1
  84. data/app/components/avo/fields/badge_field/show_component.html.erb +6 -1
  85. data/app/components/avo/fields/belongs_to_field/edit_component.html.erb +16 -14
  86. data/app/components/avo/fields/belongs_to_field/edit_component.rb +13 -3
  87. data/app/components/avo/fields/boolean_field/edit_component.html.erb +2 -2
  88. data/app/components/avo/fields/boolean_group_field/edit_component.html.erb +1 -1
  89. data/app/components/avo/fields/boolean_group_field/edit_component.rb +2 -3
  90. data/app/components/avo/fields/code_field/show_component.html.erb +1 -1
  91. data/app/components/avo/fields/common/boolean_check_component.rb +3 -3
  92. data/app/components/avo/fields/common/boolean_group_component.html.erb +2 -2
  93. data/app/components/avo/fields/common/files/controls_component.html.erb +30 -27
  94. data/app/components/avo/fields/common/files/controls_component.rb +0 -1
  95. data/app/components/avo/fields/common/files/list_viewer_component.html.erb +4 -5
  96. data/app/components/avo/fields/common/files/view_type/grid_item_component.html.erb +1 -1
  97. data/app/components/avo/fields/common/files/view_type/list_item_component.html.erb +7 -19
  98. data/app/components/avo/fields/common/heading_component.html.erb +2 -2
  99. data/app/components/avo/fields/common/heading_component.rb +1 -1
  100. data/app/components/avo/fields/common/key_value_component.html.erb +20 -24
  101. data/app/components/avo/fields/common/nested_field_component.html.erb +1 -1
  102. data/app/components/avo/fields/common/nested_field_component.rb +1 -1
  103. data/app/components/avo/fields/common/progress_bar_component.html.erb +2 -2
  104. data/app/components/avo/fields/common/stars_component.html.erb +1 -1
  105. data/app/components/avo/fields/common/status_viewer_component.html.erb +4 -10
  106. data/app/components/avo/fields/date_field/edit_component.html.erb +14 -1
  107. data/app/components/avo/fields/date_time_field/edit_component.html.erb +3 -3
  108. data/app/components/avo/fields/easy_mde_field/show_component.html.erb +1 -1
  109. data/app/components/avo/fields/edit_component.rb +9 -5
  110. data/app/components/avo/fields/file_field/edit_component.html.erb +14 -6
  111. data/app/components/avo/fields/files_field/edit_component.html.erb +11 -4
  112. data/app/components/avo/fields/has_one_field/show_component.html.erb +10 -9
  113. data/app/components/avo/fields/heading_field/edit_component.html.erb +1 -1
  114. data/app/components/avo/fields/heading_field/show_component.html.erb +2 -2
  115. data/app/components/avo/fields/password_field/edit_component.html.erb +5 -7
  116. data/app/components/avo/fields/progress_bar_field/edit_component.html.erb +1 -1
  117. data/app/components/avo/fields/radio_field/edit_component.html.erb +1 -1
  118. data/app/components/avo/fields/show_component.rb +4 -7
  119. data/app/components/avo/fields/stars_field/edit_component.html.erb +3 -4
  120. data/app/components/avo/fields/tags_field/edit_component.html.erb +1 -1
  121. data/app/components/avo/fields/tags_field/tag_component.html.erb +7 -8
  122. data/app/components/avo/fields/time_field/edit_component.html.erb +3 -3
  123. data/app/components/avo/fields/tiptap_field/edit_component.html.erb +11 -11
  124. data/app/components/avo/fields/tiptap_field/show_component.html.erb +4 -4
  125. data/app/components/avo/fields/trix_field/edit_component.html.erb +2 -2
  126. data/app/components/avo/filters_component.html.erb +8 -16
  127. data/app/components/avo/index/field_wrapper_component.html.erb +9 -2
  128. data/app/components/avo/index/field_wrapper_component.rb +0 -16
  129. data/app/components/avo/index/grid_cover_empty_state_component.html.erb +2 -2
  130. data/app/components/avo/index/grid_item_component.html.erb +29 -12
  131. data/app/components/avo/index/grid_item_component.rb +12 -10
  132. data/app/components/avo/index/resource_controls_component.html.erb +1 -1
  133. data/app/components/avo/index/resource_controls_component.rb +7 -17
  134. data/app/components/avo/index/resource_controls_dropdown_component.html.erb +3 -0
  135. data/app/components/avo/index/resource_controls_dropdown_component.rb +81 -0
  136. data/app/components/avo/index/table_row_component.html.erb +11 -11
  137. data/app/components/avo/items/panel_component.html.erb +8 -18
  138. data/app/components/avo/items/panel_component.rb +0 -20
  139. data/app/components/avo/items/switcher_component.html.erb +62 -1
  140. data/app/components/avo/items/switcher_component.rb +10 -5
  141. data/app/components/avo/items/visible_items_component.html.erb +2 -1
  142. data/app/components/avo/items/visible_items_component.rb +1 -0
  143. data/app/components/avo/media_library/item_details_component.html.erb +2 -2
  144. data/app/components/avo/media_library/list_component.html.erb +24 -19
  145. data/app/components/avo/media_library/list_component.rb +2 -2
  146. data/app/components/avo/media_library/list_item_component.html.erb +3 -3
  147. data/app/components/avo/modal_component.html.erb +52 -20
  148. data/app/components/avo/modal_component.rb +7 -12
  149. data/app/components/avo/paginator_component.html.erb +46 -33
  150. data/app/components/avo/paginator_component.rb +10 -5
  151. data/app/components/avo/panel_name_component.html.erb +1 -1
  152. data/app/components/avo/profile_item_component.html.erb +4 -4
  153. data/app/components/avo/profile_item_component.rb +2 -4
  154. data/app/components/avo/referrer_params_component.html.erb +1 -1
  155. data/app/components/avo/resource_component.rb +8 -19
  156. data/app/components/avo/resource_listing_component.html.erb +22 -0
  157. data/app/components/avo/resource_listing_component.rb +28 -0
  158. data/app/components/avo/resource_sidebar_component.html.erb +2 -2
  159. data/app/components/avo/resource_sidebar_component.rb +0 -4
  160. data/app/components/avo/row_component.html.erb +1 -1
  161. data/app/components/avo/row_selector_component.html.erb +2 -4
  162. data/app/components/avo/row_selector_component.rb +0 -1
  163. data/app/components/avo/search_overlay_component.html.erb +6 -0
  164. data/app/components/avo/search_overlay_component.rb +2 -0
  165. data/app/components/avo/sidebar/base_item_component.rb +9 -9
  166. data/app/components/avo/sidebar/group_component.html.erb +23 -27
  167. data/app/components/avo/sidebar/group_component.rb +4 -0
  168. data/app/components/avo/sidebar/link_component.html.erb +30 -7
  169. data/app/components/avo/sidebar/link_component.rb +30 -2
  170. data/app/components/avo/sidebar/section_component.html.erb +31 -11
  171. data/app/components/avo/sidebar/section_component.rb +1 -3
  172. data/app/components/avo/sidebar_component.html.erb +35 -36
  173. data/app/components/avo/sidebar_component.rb +16 -0
  174. data/app/components/avo/sidebar_profile_component.html.erb +27 -36
  175. data/app/components/avo/sidebar_profile_component.rb +14 -0
  176. data/app/components/avo/tab_group_component.html.erb +21 -9
  177. data/app/components/avo/tab_group_component.rb +54 -15
  178. data/app/components/avo/turbo_frame_wrapper_component.html.erb +4 -8
  179. data/app/components/avo/turbo_frame_wrapper_component.rb +9 -0
  180. data/app/components/avo/u_i/avatar_component.html.erb +11 -0
  181. data/app/components/avo/u_i/avatar_component.rb +61 -0
  182. data/app/components/avo/u_i/badge_component.html.erb +9 -0
  183. data/app/components/avo/u_i/badge_component.rb +35 -0
  184. data/app/components/avo/u_i/card_component.html.erb +29 -0
  185. data/app/components/avo/u_i/card_component.rb +65 -0
  186. data/app/components/avo/u_i/dropdown_component.html.erb +14 -0
  187. data/app/components/avo/u_i/dropdown_component.rb +26 -0
  188. data/app/components/avo/u_i/dropdown_menu_component.html.erb +16 -0
  189. data/app/components/avo/u_i/dropdown_menu_component.rb +8 -0
  190. data/app/components/avo/u_i/file_upload_input_component.html.erb +26 -0
  191. data/app/components/avo/u_i/file_upload_input_component.rb +15 -0
  192. data/app/components/avo/u_i/file_upload_item_component.html.erb +48 -0
  193. data/app/components/avo/u_i/file_upload_item_component.rb +17 -0
  194. data/app/components/avo/u_i/icon_button_component.html.erb +7 -0
  195. data/app/components/avo/u_i/icon_button_component.rb +22 -0
  196. data/app/components/avo/u_i/panel_component.html.erb +49 -0
  197. data/app/components/avo/u_i/panel_component.rb +29 -0
  198. data/app/components/avo/u_i/panel_header_component.html.erb +37 -0
  199. data/app/components/avo/u_i/panel_header_component.rb +24 -0
  200. data/app/components/avo/u_i/search_input_component.html.erb +28 -0
  201. data/app/components/avo/u_i/search_input_component.rb +12 -0
  202. data/app/components/avo/u_i/tabs/tab_component.html.erb +26 -0
  203. data/app/components/avo/u_i/tabs/tab_component.rb +25 -0
  204. data/app/components/avo/u_i/tabs/tabs_component.html.erb +7 -0
  205. data/app/components/avo/u_i/tabs/tabs_component.rb +13 -0
  206. data/app/components/avo/view_types/base_view_type_component.rb +26 -0
  207. data/app/components/avo/view_types/grid_component.html.erb +18 -0
  208. data/app/components/avo/view_types/grid_component.rb +4 -0
  209. data/app/components/avo/view_types/map_component.html.erb +17 -0
  210. data/app/components/avo/view_types/map_component.rb +110 -0
  211. data/app/components/avo/view_types/table_component.html.erb +63 -0
  212. data/app/components/avo/{index/resource_table_component.rb → view_types/table_component.rb} +1 -10
  213. data/app/components/avo/views/resource_edit_component.html.erb +12 -18
  214. data/app/components/avo/views/resource_edit_component.rb +1 -27
  215. data/app/components/avo/views/resource_index_component.html.erb +97 -74
  216. data/app/components/avo/views/resource_index_component.rb +21 -1
  217. data/app/components/avo/views/resource_show_component.html.erb +4 -2
  218. data/app/components/avo/views/resource_show_component.rb +1 -4
  219. data/app/controllers/avo/associations_controller.rb +1 -1
  220. data/app/controllers/avo/base_application_controller.rb +7 -5
  221. data/app/controllers/avo/base_controller.rb +139 -40
  222. data/app/controllers/avo/debug_controller.rb +0 -29
  223. data/app/controllers/avo/media_library_controller.rb +17 -1
  224. data/app/controllers/avo/search_controller.rb +5 -0
  225. data/app/controllers/concerns/avo/initializes_avo.rb +7 -8
  226. data/app/helpers/avo/application_helper.rb +60 -46
  227. data/app/helpers/avo/turbo_stream_actions_helper.rb +10 -1
  228. data/app/helpers/avo/url_helpers.rb +3 -2
  229. data/app/javascript/{avo.base.js → application.js} +9 -18
  230. data/app/javascript/js/application.js +40 -0
  231. data/app/javascript/js/controllers/action_controller.js +4 -4
  232. data/app/javascript/js/controllers/actions_overflow_controller.js +21 -6
  233. data/app/javascript/js/controllers/color_scheme_switcher_controller.js +226 -0
  234. data/app/javascript/js/controllers/dropdown_menu_controller.js +42 -0
  235. data/app/javascript/js/controllers/fields/code_field_controller.js +20 -3
  236. data/app/javascript/js/controllers/fields/easy_mde_controller.js +23 -1
  237. data/app/javascript/js/controllers/fields/key_value_controller.js +43 -39
  238. data/app/javascript/js/controllers/fields/panel_refresh_controller.js +4 -0
  239. data/app/javascript/js/controllers/fields/tags_field_controller.js +2 -2
  240. data/app/javascript/js/controllers/fields/tags_field_helpers.js +4 -6
  241. data/app/javascript/js/controllers/item_selector_controller.js +0 -2
  242. data/app/javascript/js/controllers/loading_button_controller.js +1 -5
  243. data/app/javascript/js/controllers/map_dark_mode_controller.js +131 -0
  244. data/app/javascript/js/controllers/menu_controller.js +38 -16
  245. data/app/javascript/js/controllers/modal_controller.js +16 -0
  246. data/app/javascript/js/controllers/modal_size_controller.js +83 -0
  247. data/app/javascript/js/controllers/nested_form_controller.js +2 -2
  248. data/app/javascript/js/controllers/password_visibility_controller.js +13 -0
  249. data/app/javascript/js/controllers/preview_controller.js +2 -2
  250. data/app/javascript/js/controllers/resource_search_controller.js +123 -0
  251. data/app/javascript/js/controllers/search_controller.js +10 -29
  252. data/app/javascript/js/controllers/sidebar_controller.js +29 -9
  253. data/app/javascript/js/controllers/table_row_controller.js +28 -0
  254. data/app/javascript/js/controllers/tippy_controller.js +1 -1
  255. data/app/javascript/js/controllers/toggle_controller.js +40 -5
  256. data/app/javascript/js/controllers.js +12 -2
  257. data/app/javascript/js/custom-stream-actions.js +10 -8
  258. data/app/views/avo/actions/show.html.erb +14 -10
  259. data/app/views/avo/base/_boolean_filter.html.erb +1 -1
  260. data/app/views/avo/base/_date_time_filter.html.erb +3 -3
  261. data/app/views/avo/base/_multiple_select_filter.html.erb +1 -1
  262. data/app/views/avo/base/_new_via_belongs_to.html.erb +2 -3
  263. data/app/views/avo/base/_text_filter.html.erb +1 -1
  264. data/app/views/avo/base/preview.html.erb +5 -4
  265. data/app/views/avo/debug/status.html.erb +5 -92
  266. data/app/views/avo/debug/status.text.erb +0 -2
  267. data/app/views/avo/home/_actions.html.erb +1 -1
  268. data/app/views/avo/home/_dashboards.html.erb +1 -1
  269. data/app/views/avo/home/_filters.html.erb +1 -1
  270. data/app/views/avo/home/_resources.html.erb +2 -2
  271. data/app/views/avo/home/failed_to_load.html.erb +2 -2
  272. data/app/views/avo/home/index.html.erb +25 -25
  273. data/app/views/avo/media_library/_form.html.erb +43 -28
  274. data/app/views/avo/media_library/show.html.erb +7 -3
  275. data/app/views/avo/modal/_size_selector.html.erb +46 -0
  276. data/app/views/avo/partials/_alerts.html.erb +1 -1
  277. data/app/views/avo/partials/_color_scheme_switcher.html.erb +106 -0
  278. data/app/views/avo/partials/_color_theme_override.html.erb +49 -0
  279. data/app/views/avo/partials/_confirm_dialog.html.erb +4 -4
  280. data/app/views/avo/partials/_custom_tools_alert.html.erb +6 -6
  281. data/app/views/avo/partials/_footer.html.erb +1 -1
  282. data/app/views/avo/partials/_header.html.erb +1 -1
  283. data/app/views/avo/partials/_javascript.html.erb +0 -2
  284. data/app/views/avo/partials/_navbar.html.erb +53 -8
  285. data/app/views/avo/partials/_sortable_component.html.erb +1 -1
  286. data/app/views/avo/partials/_table_header.html.erb +28 -19
  287. data/app/views/avo/partials/_view_toggle_button.html.erb +6 -29
  288. data/app/views/avo/partials/distribution_chart.html.erb +2 -2
  289. data/app/views/avo/private/_links_and_buttons.html.erb +12 -8
  290. data/app/views/avo/private/design.html.erb +8 -4
  291. data/app/views/avo/sidebar/_license_warning.html.erb +3 -3
  292. data/app/views/layouts/avo/application.html.erb +39 -17
  293. data/app/views/layouts/avo/modal.html.erb +7 -0
  294. data/avo.gemspec +3 -4
  295. data/config/i18n-tasks.yml +1 -1
  296. data/config/importmap.rb +1 -0
  297. data/config/initializers/pagy.rb +5 -25
  298. data/config/routes/dynamic_routes.rb +4 -0
  299. data/config/routes.rb +6 -7
  300. data/db/factories.rb +14 -0
  301. data/lib/avo/asset_manager.rb +2 -0
  302. data/lib/avo/avatar.rb +7 -0
  303. data/lib/avo/base_action.rb +16 -4
  304. data/lib/avo/concerns/breadcrumbs.rb +7 -66
  305. data/lib/avo/concerns/form_builder.rb +41 -0
  306. data/lib/avo/concerns/{has_profile_photo.rb → has_avatar.rb} +8 -4
  307. data/lib/avo/concerns/{has_cover_photo.rb → has_cover.rb} +4 -4
  308. data/lib/avo/concerns/has_description.rb +9 -0
  309. data/lib/avo/concerns/has_item_type.rb +6 -2
  310. data/lib/avo/concerns/has_items.rb +43 -58
  311. data/lib/avo/concerns/pagination.rb +29 -19
  312. data/lib/avo/concerns/row_controls_configuration.rb +22 -15
  313. data/lib/avo/configuration/branding.rb +7 -7
  314. data/lib/avo/configuration.rb +92 -45
  315. data/lib/avo/{cover_photo.rb → cover.rb} +2 -2
  316. data/lib/avo/current.rb +0 -11
  317. data/lib/avo/discreet_information.rb +52 -29
  318. data/lib/avo/dsl/field_parser.rb +1 -1
  319. data/lib/avo/engine.rb +41 -8
  320. data/lib/avo/error_manager.rb +1 -1
  321. data/lib/avo/fields/avatar_field.rb +19 -0
  322. data/lib/avo/fields/badge_field.rb +23 -3
  323. data/lib/avo/fields/base_field.rb +46 -1
  324. data/lib/avo/fields/concerns/dom_id.rb +17 -0
  325. data/lib/avo/fields/concerns/file_authorization.rb +4 -0
  326. data/lib/avo/fields/concerns/has_html_attributes.rb +1 -1
  327. data/lib/avo/fields/concerns/is_searchable.rb +2 -5
  328. data/lib/avo/fields/concerns/nested.rb +1 -1
  329. data/lib/avo/fields/files_field.rb +2 -2
  330. data/lib/avo/fields/frame_base_field.rb +2 -2
  331. data/lib/avo/fields/id_field.rb +5 -1
  332. data/lib/avo/fields/progress_bar_field.rb +1 -1
  333. data/lib/avo/fields/text_field.rb +1 -1
  334. data/lib/avo/plugin_manager.rb +4 -0
  335. data/lib/avo/resources/base.rb +33 -21
  336. data/lib/avo/resources/controls/actions_list.rb +3 -3
  337. data/lib/avo/resources/controls/base_control.rb +1 -1
  338. data/lib/avo/resources/items/card.rb +16 -0
  339. data/lib/avo/resources/items/header.rb +11 -0
  340. data/lib/avo/resources/items/holder.rb +18 -23
  341. data/lib/avo/resources/items/item_group.rb +5 -7
  342. data/lib/avo/resources/items/sidebar.rb +5 -9
  343. data/lib/avo/resources/items/tab.rb +8 -8
  344. data/lib/avo/resources/items/tab_group.rb +8 -10
  345. data/lib/avo/resources/resource_manager.rb +2 -1
  346. data/lib/avo/services/hq_reporter.rb +102 -0
  347. data/lib/avo/services/telemetry_service.rb +0 -1
  348. data/lib/avo/test_helpers.rb +36 -22
  349. data/lib/avo/u_i_instance.rb +60 -0
  350. data/lib/avo/version.rb +1 -1
  351. data/lib/avo/view_inquirer.rb +6 -1
  352. data/lib/avo/view_types/view_type_manager.rb +70 -0
  353. data/lib/avo.rb +30 -45
  354. data/lib/generators/avo/action_generator.rb +1 -1
  355. data/lib/generators/avo/resource_generator.rb +43 -0
  356. data/lib/generators/avo/resource_tool_generator.rb +1 -1
  357. data/lib/generators/avo/tailwindcss/install_generator.rb +0 -6
  358. data/lib/generators/avo/templates/cards/partial_card_partial.tt +1 -1
  359. data/lib/generators/avo/templates/initializer/avo.tt +7 -9
  360. data/lib/generators/avo/templates/locales/avo.ar.yml +25 -0
  361. data/lib/generators/avo/templates/locales/avo.de.yml +25 -0
  362. data/lib/generators/avo/templates/locales/avo.en.yml +25 -0
  363. data/lib/generators/avo/templates/locales/avo.es.yml +25 -0
  364. data/lib/generators/avo/templates/locales/avo.fr.yml +25 -0
  365. data/lib/generators/avo/templates/locales/avo.it.yml +25 -0
  366. data/lib/generators/avo/templates/locales/avo.ja.yml +25 -0
  367. data/lib/generators/avo/templates/locales/avo.nb.yml +25 -0
  368. data/lib/generators/avo/templates/locales/avo.nl.yml +25 -0
  369. data/lib/generators/avo/templates/locales/avo.nn.yml +25 -0
  370. data/lib/generators/avo/templates/locales/avo.pl.yml +25 -0
  371. data/lib/generators/avo/templates/locales/avo.pt-BR.yml +25 -0
  372. data/lib/generators/avo/templates/locales/avo.pt.yml +25 -0
  373. data/lib/generators/avo/templates/locales/avo.ro.yml +25 -0
  374. data/lib/generators/avo/templates/locales/avo.ru.yml +25 -0
  375. data/lib/generators/avo/templates/locales/avo.tr.yml +25 -0
  376. data/lib/generators/avo/templates/locales/{avo.uk.yml → avo.ua.yml} +26 -1
  377. data/lib/generators/avo/templates/locales/avo.zh-TW.yml +25 -0
  378. data/lib/generators/avo/templates/locales/avo.zh.yml +25 -0
  379. data/lib/generators/avo/templates/resource/resource.tt +10 -1
  380. data/lib/generators/avo/templates/resource_tools/partial.tt +20 -22
  381. data/lib/generators/avo/templates/tailwindcss/avo.tailwind.css +1 -3
  382. data/lib/generators/avo/templates/tailwindcss/tailwind.config.js +0 -7
  383. data/lib/generators/avo/templates/tool/view.tt +14 -16
  384. data/lib/generators/avo/tool_generator.rb +3 -3
  385. data/lib/generators/avo/version_generator.rb +1 -1
  386. data/lib/tasks/avo_tasks.rake +29 -25
  387. metadata +144 -97
  388. data/app/assets/stylesheets/avo.base.css +0 -130
  389. data/app/assets/stylesheets/css/breadcrumbs.css +0 -16
  390. data/app/assets/stylesheets/css/buttons.css +0 -19
  391. data/app/assets/stylesheets/css/tailwindcss/base.css +0 -1
  392. data/app/assets/stylesheets/css/tailwindcss/components.css +0 -1
  393. data/app/assets/stylesheets/css/tailwindcss/utilities.css +0 -1
  394. data/app/assets/svgs/avo/arrow-down.svg +0 -3
  395. data/app/assets/svgs/avo/arrow-up.svg +0 -3
  396. data/app/assets/svgs/avo/detach.svg +0 -8
  397. data/app/assets/svgs/avo/download-solid-reversed.svg +0 -3
  398. data/app/assets/svgs/avo/download-solid.svg +0 -3
  399. data/app/assets/svgs/avo/edit.svg +0 -5
  400. data/app/assets/svgs/avo/editor-bold.svg +0 -1
  401. data/app/assets/svgs/avo/editor-italic.svg +0 -1
  402. data/app/assets/svgs/avo/editor-link.svg +0 -1
  403. data/app/assets/svgs/avo/editor-list.svg +0 -1
  404. data/app/assets/svgs/avo/editor-strike.svg +0 -1
  405. data/app/assets/svgs/avo/editor-underline.svg +0 -1
  406. data/app/assets/svgs/avo/eye.svg +0 -1
  407. data/app/assets/svgs/avo/trash-sm.svg +0 -3
  408. data/app/assets/svgs/avo/trash.svg +0 -7
  409. data/app/components/avo/cover_photo_component.html.erb +0 -3
  410. data/app/components/avo/cover_photo_component.rb +0 -19
  411. data/app/components/avo/fields/common/badge_viewer_component.html.erb +0 -1
  412. data/app/components/avo/fields/common/badge_viewer_component.rb +0 -33
  413. data/app/components/avo/index/resource_grid_component.html.erb +0 -23
  414. data/app/components/avo/index/resource_grid_component.rb +0 -10
  415. data/app/components/avo/index/resource_map_component.html.erb +0 -16
  416. data/app/components/avo/index/resource_map_component.rb +0 -114
  417. data/app/components/avo/index/resource_table_component.html.erb +0 -52
  418. data/app/components/avo/panel_component.html.erb +0 -63
  419. data/app/components/avo/panel_component.rb +0 -43
  420. data/app/components/avo/panel_header_component.html.erb +0 -42
  421. data/app/components/avo/panel_header_component.rb +0 -31
  422. data/app/components/avo/profile_photo_component.html.erb +0 -6
  423. data/app/components/avo/profile_photo_component.rb +0 -9
  424. data/app/components/avo/sidebar/heading_component.html.erb +0 -21
  425. data/app/components/avo/sidebar/heading_component.rb +0 -9
  426. data/app/components/avo/tab_switcher_component.html.erb +0 -20
  427. data/app/components/avo/tab_switcher_component.rb +0 -45
  428. data/app/views/avo/base/close_modal_and_reload_field.turbo_stream.erb +0 -8
  429. data/app/views/avo/debug/report.html.erb +0 -37
  430. data/app/views/avo/partials/_panel_breadcrumbs.html.erb +0 -3
  431. data/app/views/avo/partials/_resource_search.html.erb +0 -16
  432. data/lib/avo/licensing/community_license.rb +0 -6
  433. data/lib/avo/licensing/h_q.rb +0 -202
  434. data/lib/avo/licensing/license.rb +0 -76
  435. data/lib/avo/licensing/license_manager.rb +0 -24
  436. data/lib/avo/licensing/nil_license.rb +0 -14
  437. data/lib/avo/licensing/pro_license.rb +0 -20
  438. data/lib/avo/licensing/request.rb +0 -20
  439. data/lib/avo/profile_photo.rb +0 -7
  440. data/lib/avo/services/debug_service.rb +0 -107
  441. data/tailwind.preset.js +0 -171
  442. /data/{public/avo-assets → app/assets/images/avo}/favicon.ico +0 -0
  443. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.eot +0 -0
  444. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.svg +0 -0
  445. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.ttf +0 -0
  446. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.woff +0 -0
  447. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-500.woff2 +0 -0
  448. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.eot +0 -0
  449. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.svg +0 -0
  450. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.ttf +0 -0
  451. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.woff +0 -0
  452. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-600.woff2 +0 -0
  453. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.eot +0 -0
  454. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.svg +0 -0
  455. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.ttf +0 -0
  456. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.woff +0 -0
  457. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-700.woff2 +0 -0
  458. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.eot +0 -0
  459. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.svg +0 -0
  460. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.ttf +0 -0
  461. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.woff +0 -0
  462. /data/{public/avo-assets → app/assets/images/avo}/fonts/inter-v7-latin-regular.woff2 +0 -0
  463. /data/{public/avo-assets → app/assets/images/avo}/logo-on-white.png +0 -0
  464. /data/{public/avo-assets → app/assets/images/avo}/logo.png +0 -0
  465. /data/{public/avo-assets → app/assets/images/avo}/logomark.png +0 -0
  466. /data/{public/avo-assets → app/assets/images/avo}/placeholder.svg +0 -0
@@ -0,0 +1,14 @@
1
+ <%= content_tag :div, class: class_names("dropdown"), data: do %>
2
+ <%= trigger %>
3
+
4
+ <% if items.present? %>
5
+ <%= render Avo::UI::DropdownMenuComponent.new(
6
+ data: {dropdown_menu_target: "menu"}.merge(@data),
7
+ classes: @classes,
8
+ dropdown_menu_classes: @dropdown_menu_classes,
9
+ open: @open
10
+ ) do %>
11
+ <%= items %>
12
+ <% end %>
13
+ <% end %>
14
+ <% end %>
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::UI::DropdownComponent < Avo::BaseComponent
4
+ prop :classes
5
+ prop :data, default: {}.freeze
6
+ prop :wrapper_data, default: {}.freeze
7
+ prop :open, default: false
8
+ prop :dropdown_menu_classes, default: ""
9
+
10
+ renders_one :trigger
11
+ renders_one :items
12
+
13
+ # this is used to trigger the dropdown menu from trigger element
14
+ # data: {action: component.action} => click->dropdown-menu#toggle
15
+ def action
16
+ @action ||= "click->dropdown-menu#toggle"
17
+ end
18
+
19
+ def data
20
+ return {} if items.blank?
21
+
22
+ {
23
+ controller: "dropdown-menu",
24
+ }.merge(@wrapper_data)
25
+ end
26
+ end
@@ -0,0 +1,16 @@
1
+ <%= content_tag :dialog,
2
+ class: class_names(
3
+ "dropdown-popover",
4
+ "css-animate-dialog-slide-down",
5
+ @classes
6
+ ),
7
+ open: @open,
8
+ data: @data do %>
9
+ <div class="dropdown-menu <%= @dropdown_menu_classes %>">
10
+ <div class="dropdown-menu__group">
11
+ <div class="dropdown-menu__list">
12
+ <%= content %>
13
+ </div>
14
+ </div>
15
+ </div>
16
+ <% end %>
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::UI::DropdownMenuComponent < Avo::BaseComponent
4
+ prop :data, default: {}.freeze
5
+ prop :classes
6
+ prop :open, default: false
7
+ prop :dropdown_menu_classes, default: ""
8
+ end
@@ -0,0 +1,26 @@
1
+ <%= content_tag :label, for: @id, class: class_names(
2
+ "file-upload-input",
3
+ "file-upload-input--#{@size}",
4
+ {
5
+ "file-upload-input--disabled": @disabled
6
+ },
7
+ @classes) do %>
8
+ <span class="file-upload-input__icon" aria-hidden="true">
9
+ <%= svg "tabler/outline/upload" %>
10
+ </span>
11
+
12
+ <% if @size != :sm %>
13
+ <span class="file-upload-input__content">
14
+ <span class="file-upload-input__title">
15
+ <%= @title %>
16
+ <span class="file-upload-input__subtitle"><%= @subtitle %></span>
17
+ </span>
18
+
19
+ <%= content_tag(:span, @description, class: "file-upload-input__description") %>
20
+ </span>
21
+ <% end %>
22
+
23
+ <div class="file-upload-input__native-input">
24
+ <%= content if content.present? %>
25
+ </div>
26
+ <% end %>
@@ -0,0 +1,15 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::UI::FileUploadInputComponent < Avo::BaseComponent
4
+ VALID_SIZES = %i[sm md lg].freeze unless defined?(VALID_SIZES)
5
+
6
+ prop :id
7
+ prop :title, default: "Click to upload"
8
+ prop :subtitle, default: "or drag and drop"
9
+ prop :description
10
+ prop :size, default: :lg do |value|
11
+ VALID_SIZES.include?(value) ? value : :lg
12
+ end
13
+ prop :disabled, default: false
14
+ prop :classes
15
+ end
@@ -0,0 +1,48 @@
1
+ <%= content_tag :div, class: class_names("file-upload-item",{
2
+ "file-upload-item--uploading": show_progress?,
3
+ }, @classes) do %>
4
+ <span
5
+ class="file-upload-item__progress"
6
+ style="width: <%= progress_percentage %>%;"
7
+ aria-hidden="true">
8
+ </span>
9
+
10
+ <div class="file-upload-item__body">
11
+ <p class="file-upload-item__title" title="<%= @title %>"><%= @title %></p>
12
+
13
+ <div class="file-upload-item__meta">
14
+ <span class="file-upload-item__size">
15
+ <% if @state == :error || @state == :failed %>
16
+ Upload failed.
17
+ <% else %>
18
+ <%= @size_label %>
19
+ <% end %>
20
+ </span>
21
+
22
+ <% if show_progress? %>
23
+ <span aria-hidden="true">|</span>
24
+
25
+ <% if @state == :complete %>
26
+ <span class="file-upload-item__status file-upload-item__status--complete">
27
+ <%= svg("tabler/outline/circle-check", class: "file-upload-item__status-icon") %>
28
+ <span class="file-upload-item__status-label">Complete</span>
29
+ </span>
30
+ <% end %>
31
+ <% if @state == :error || @state == :failed %>
32
+ <span class="file-upload-item__status file-upload-item__status--error">
33
+ <%= svg("tabler/outline/circle-x", class: "file-upload-item__status-icon") %>
34
+ <span class="file-upload-item__status-label">Try again</span>
35
+ </span>
36
+ <% end %>
37
+ <% if @state == :uploading %>
38
+ <span class="file-upload-item__status file-upload-item__status--uploading">
39
+ <%= svg("tabler/outline/upload", class: "file-upload-item__status-icon") %>
40
+ <span class="file-upload-item__status-label"><%= progress_percentage %>%</span>
41
+ </span>
42
+ <% end %>
43
+ <% end %>
44
+ </div>
45
+ </div>
46
+
47
+ <%= content_tag :div, content, class: "file-upload-item__actions" if content.present? %>
48
+ <% end %>
@@ -0,0 +1,17 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::UI::FileUploadItemComponent < Avo::BaseComponent
4
+ prop :title
5
+ prop :size_label
6
+ prop :progress, default: 0
7
+ prop :classes
8
+ prop :state, default: :default # :uploading, :complete, :error, :failed after we wull implement the states dynamically
9
+
10
+ def show_progress?
11
+ @progress.to_i > 0 || @state != :default
12
+ end
13
+
14
+ def progress_percentage
15
+ @progress.to_i.clamp(0, 100)
16
+ end
17
+ end
@@ -0,0 +1,7 @@
1
+ <%= tag.button(
2
+ type: @type,
3
+ class: button_classes,
4
+ data: data
5
+ ) do %>
6
+ <%= svg(@icon) %>
7
+ <% end %>
@@ -0,0 +1,22 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::UI::IconButtonComponent < Avo::BaseComponent
4
+ prop :icon
5
+ prop :action
6
+ prop :classes
7
+ prop :type, default: :button
8
+ prop :data, default: {}.freeze
9
+
10
+ def data
11
+ {
12
+ action: @action
13
+ }.merge(@data)
14
+ end
15
+
16
+ def button_classes
17
+ class_names(
18
+ "inline-flex items-center justify-center rounded-md p-1 cursor-pointer",
19
+ @classes
20
+ )
21
+ end
22
+ end
@@ -0,0 +1,49 @@
1
+ <%= tag.div class: class_names(
2
+ "panel",
3
+ @class,
4
+ "panel--has-sidebar": sidebar?,
5
+ "has-cover": has_cover?,
6
+ ), data: {
7
+ **@data,
8
+ component_name:,
9
+ "item-index": @index
10
+ } do %>
11
+ <% if cover? %>
12
+ <div class="panel__cover"><%= cover %></div>
13
+ <% end %>
14
+
15
+ <% if header? %>
16
+ <div class="panel__header"><%= header %></div>
17
+ <% elsif title.present? || description.present? || controls? %>
18
+ <div class="panel__header">
19
+ <%= render ui.panel_header(
20
+ title:,
21
+ description:,
22
+ ) do |header| %>
23
+ <% if controls? %>
24
+ <% header.with_controls do %>
25
+ <%= controls %>
26
+ <% end %>
27
+ <% end %>
28
+ <% end %>
29
+ </div>
30
+ <% end %>
31
+
32
+ <div class="panel__content">
33
+ <div class="panel__body">
34
+ <% if card? %>
35
+ <div class="panel__card"><%= card %></div>
36
+ <% else %>
37
+ <%= body %>
38
+ <% end %>
39
+
40
+ <% if footer? %>
41
+ <div class="panel__footer"><%= footer %></div>
42
+ <% end %>
43
+ </div>
44
+
45
+ <% if sidebar? %>
46
+ <div class="panel__sidebar"><%= sidebar %></div>
47
+ <% end %>
48
+ </div>
49
+ <% end %>
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::UI::PanelComponent < Avo::BaseComponent
4
+ prop :title, reader: :public
5
+ prop :description, reader: :public
6
+ prop :external_link
7
+ prop :args, kind: :**, default: {}.freeze
8
+ prop :avatar
9
+ prop :cover
10
+ prop :class
11
+ prop :index
12
+ prop :data, default: -> { {}.freeze }
13
+
14
+ renders_one :header
15
+ renders_one :controls
16
+ renders_one :cover
17
+ renders_one :sidebar
18
+ renders_one :body
19
+ renders_one :card, "Avo::UI::CardComponent" # wraps content into a card automatically
20
+ renders_one :footer
21
+
22
+ def has_cover?
23
+ @cover.present?
24
+ end
25
+
26
+ def has_avatar?
27
+ @avatar.present?
28
+ end
29
+ end
@@ -0,0 +1,37 @@
1
+ <%= tag.div class: class_names(
2
+ @class,
3
+ "header",
4
+ "header--size-medium": size_medium?,
5
+ "header--size-small": size_small?
6
+ ), data: { "item-index": @index }.compact_blank do %>
7
+ <div class="header__main">
8
+ <div class="header__content">
9
+ <% if avatar? %>
10
+ <div class="header__avatar"><%= avatar %></div>
11
+ <% end %>
12
+
13
+ <div class="header__text">
14
+ <div class="header__title">
15
+ <% if @url.present? %>
16
+ <%= link_to @title, @url, target: @target %>
17
+ <% else %>
18
+ <%= @title %>
19
+ <% end %>
20
+ <%= title_slot %>
21
+ </div>
22
+
23
+ <% if @description.present? %>
24
+ <p class="header__description"><%= @description %></p>
25
+ <% end %>
26
+
27
+ <% if discreet_information? %>
28
+ <div class="header__discreet-information mt-2"><%= discreet_information %></div>
29
+ <% end %>
30
+ </div>
31
+ </div>
32
+
33
+ <% if controls? %>
34
+ <div class="header__controls"><%= controls %></div>
35
+ <% end %>
36
+ </div>
37
+ <% end %>
@@ -0,0 +1,24 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::UI::PanelHeaderComponent < Avo::BaseComponent
4
+ prop :title
5
+ prop :description
6
+ prop :size, default: :medium
7
+ prop :url
8
+ prop :target
9
+ prop :index
10
+ prop :class
11
+
12
+ renders_one :title_slot
13
+ renders_one :avatar
14
+ renders_one :controls
15
+ renders_one :discreet_information
16
+
17
+ def size_medium?
18
+ @size == :medium
19
+ end
20
+
21
+ def size_small?
22
+ @size == :small
23
+ end
24
+ end
@@ -0,0 +1,28 @@
1
+ <%= content_tag :div, class: "search-input" do %>
2
+ <%= search_field_tag @name, @value,
3
+ id: @id,
4
+ placeholder: @placeholder,
5
+ class: class_names(
6
+ "search-input__input",
7
+ {"search-input__input--with-shortcut": @with_shortcut},
8
+ @classes
9
+ ),
10
+ disabled: @disabled,
11
+ autocomplete: "off",
12
+ data: @data
13
+ %>
14
+
15
+ <span class="search-input__prefix" aria-hidden="true">
16
+ <%= helpers.svg "tabler/outline/search" %>
17
+ </span>
18
+
19
+ <% if @with_shortcut %>
20
+ <span class="search-input__suffix" aria-hidden="true">
21
+ <kbd class="search-input__shortcut">
22
+ <abbr title="Command" class="no-underline os-pc:hidden">⌘</abbr>
23
+ <abbr title="CTRL" class="no-underline os-mac:hidden">CTRL</abbr>
24
+ </kbd>
25
+ <kbd class="search-input__shortcut">K</kbd>
26
+ </span>
27
+ <% end %>
28
+ <% end %>
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::UI::SearchInputComponent < Avo::BaseComponent
4
+ prop :name, default: "q"
5
+ prop :id
6
+ prop :value
7
+ prop :placeholder
8
+ prop :disabled, default: false
9
+ prop :with_shortcut, default: false
10
+ prop :classes
11
+ prop :data, default: -> { {} }
12
+ end
@@ -0,0 +1,26 @@
1
+ <%= content_tag :div, class: class_names(
2
+ "tabs__item-wrapper",
3
+ "tabs__item-wrapper--#{@variant}",
4
+ "#{ACTIVE_CLASS}": @variant == :scope && @active
5
+ ) do %>
6
+ <%= link_to(@href,
7
+ class: class_names(
8
+ @classes,
9
+ "tabs__item",
10
+ "tabs__item--group": @variant == :group,
11
+ "tabs__item--disabled": @disabled,
12
+ "#{ACTIVE_CLASS}": @active
13
+ ),
14
+ data: @data,
15
+ title: @title,
16
+ role: "tab",
17
+ id: @id,
18
+ "aria-selected": @active,
19
+ "aria-controls": @aria_controls,
20
+ "aria-disabled": @disabled,
21
+ tabindex: @disabled ? -1 : 0
22
+ ) do %>
23
+ <%= svg @icon, class: "tabs__item-icon", "aria-hidden": "true" if @icon.present? %>
24
+ <%= tag.span @label, class: "tabs__item-label" if @label.present? %>
25
+ <% end %>
26
+ <% end %>
@@ -0,0 +1,25 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Avo
4
+ module UI
5
+ module Tabs
6
+ class TabComponent < Avo::BaseComponent
7
+ include Avo::ApplicationHelper
8
+
9
+ ACTIVE_CLASS = "tabs__item--active"
10
+
11
+ prop :label
12
+ prop :variant, default: :scope
13
+ prop :active, default: false
14
+ prop :icon
15
+ prop :href, default: "#"
16
+ prop :id, default: -> { "tab-#{object_id}" }
17
+ prop :disabled, default: false
18
+ prop :aria_controls, default: -> { "#{@id}-panel" }
19
+ prop :data, default: {}.freeze
20
+ prop :classes
21
+ prop :title
22
+ end
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,7 @@
1
+ <%= content_tag :div,
2
+ class: "tabs",
3
+ role: :tablist,
4
+ id: @id,
5
+ "aria-label": @aria_label do %>
6
+ <%= content %>
7
+ <% end %>
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Avo
4
+ module UI
5
+ module Tabs
6
+ class TabsComponent < Avo::BaseComponent
7
+ prop :variant, default: :scope
8
+ prop :aria_label, default: "Tabs navigation"
9
+ prop :id, default: -> { "tabs-#{object_id}" }
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,26 @@
1
+ class Avo::ViewTypes::BaseViewTypeComponent < Avo::BaseComponent
2
+ extend PropInitializer::Properties
3
+
4
+ prop :resources
5
+ prop :resource
6
+ prop :reflection
7
+ prop :parent_record
8
+ prop :parent_resource
9
+ prop :actions, reader: :public
10
+ prop :pagy
11
+ prop :query
12
+ prop :turbo_frame
13
+ prop :index_params
14
+
15
+ def paginator_component
16
+ Avo::PaginatorComponent.new(
17
+ pagy: @pagy,
18
+ turbo_frame: @turbo_frame,
19
+ index_params: @index_params,
20
+ resource: @resource,
21
+ parent_record: @parent_record,
22
+ parent_resource: @parent_resource,
23
+ discreet_pagination: field&.discreet_pagination
24
+ )
25
+ end
26
+ end
@@ -0,0 +1,18 @@
1
+ <turbo-frame id="<%= @resource.model_key %>_list" target="_top" class="relative flex w-full flex-col">
2
+ <%= content_tag :div,
3
+ class:"grid-wrapper",
4
+ data: {
5
+ component_name: self.class.to_s.underscore,
6
+ selected_resources_name: @resource.model_key,
7
+ selected_resources: [],
8
+ **(try(:drag_reorder_data_attributes) || {})
9
+ } do %>
10
+ <% @resources.each_with_index do |resource, index| %>
11
+ <% cache_if Avo.configuration.cache_resources_on_index_view, resource.cache_hash(@parent_record) do %>
12
+ <%= render(resource.resolve_component(Avo::Index::GridItemComponent).new(resource: resource, reflection: @reflection, parent_record: @parent_record, parent_resource: @parent_resource, actions: actions)) %>
13
+ <% end %>
14
+ <% end %>
15
+ <% end %>
16
+
17
+ <div class="mt-6"><%= render paginator_component %></div>
18
+ </turbo-frame>
@@ -0,0 +1,4 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Avo::ViewTypes::GridComponent < Avo::ViewTypes::BaseViewTypeComponent
4
+ end
@@ -0,0 +1,17 @@
1
+ <% if @resources.present? %>
2
+ <div class="map-view-container grid <%= grid_layout_classes %> w-full gap-4">
3
+ <%= tag.div class: class_names("map-component rounded-2xl overflow-hidden", map_component_order_class), data: { controller: "map-dark-mode" } do %>
4
+ <%= js_map(resource_location_markers, **resource_mapkick_options) %>
5
+ <% end %>
6
+
7
+ <% if render_table? %>
8
+ <div class="overflow-auto <%= table_component_order_class %>">
9
+ <%= render(@resource.resolve_component(Avo::ViewTypes::TableComponent).new(resources: @resources, resource: @resource, reflection: @reflection, parent_record: @parent_record, parent_resource: @parent_resource, pagy: @pagy, query: @query, actions: @actions, turbo_frame: @turbo_frame, index_params: @index_params)) %>
10
+ </div>
11
+ <% end %>
12
+ </div>
13
+ <% else %>
14
+ <div class="rounded-sm bg-white shadow-panel">
15
+ <%= helpers.empty_state by_association: params[:related_name].present? %>
16
+ </div>
17
+ <% end %>
@@ -0,0 +1,110 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Render a map view for a list of resources, where each resource is
4
+ # expected to have an attribute/attribute set representing its location.
5
+ class Avo::ViewTypes::MapComponent < Avo::ViewTypes::BaseViewTypeComponent
6
+ attr_reader :resources
7
+
8
+ delegate :js_map, to: :helpers
9
+
10
+ def grid_layout_classes
11
+ return unless render_table?
12
+
13
+ if horizontal_layout?
14
+ "grid-flow-row sm:grid-flow-col grid-rows-1 auto-cols-fr"
15
+ elsif vertical_layout?
16
+ "grid-flow-row grid-cols-1"
17
+ end
18
+ end
19
+
20
+ def horizontal_layout?
21
+ %i[left right].include?(map_position)
22
+ end
23
+
24
+ def vertical_layout?
25
+ %i[bottom top].include?(map_position)
26
+ end
27
+
28
+ def map_component_order_class
29
+ if render_table? && table_positioned_at_the_start?
30
+ "order-last"
31
+ else
32
+ "order-first"
33
+ end
34
+ end
35
+
36
+ def table_component_order_class
37
+ if table_positioned_at_the_start?
38
+ "order-first"
39
+ else
40
+ "order-last"
41
+ end
42
+ end
43
+
44
+ def table_positioned_at_the_start?
45
+ %i[right bottom].include?(map_position)
46
+ end
47
+
48
+ def map_position
49
+ map_options.dig(:map, :position)
50
+ end
51
+
52
+ def resource_location_markers
53
+ # If we have no proc and no default location method, don't try to create markers
54
+ return [] unless resource_mappable?
55
+
56
+ records_markers = resources
57
+ .map do |resource|
58
+ Avo::ExecutionContext.new(target: marker_proc, record: resource.record).handle
59
+ end
60
+ .compact
61
+ .filter do |coordinates|
62
+ coordinates[:latitude].present? && coordinates[:longitude].present?
63
+ end
64
+
65
+ return records_markers if map_options[:extra_markers].nil?
66
+
67
+ records_markers + Avo::ExecutionContext.new(target: map_options[:extra_markers]).handle
68
+ end
69
+
70
+ def resource_mapkick_options
71
+ options = map_options[:mapkick_options] || {}
72
+
73
+ options[:height] = if horizontal_layout?
74
+ "100%"
75
+ else
76
+ "26rem"
77
+ end
78
+
79
+ options[:style] ||= "mapbox://styles/mapbox/light-v11"
80
+
81
+ options
82
+ end
83
+
84
+ def render_table?
85
+ map_options.dig(:table, :visible)
86
+ end
87
+
88
+ private
89
+
90
+ def default_record_marker_proc
91
+ lambda {
92
+ {
93
+ latitude: record.coordinates.first,
94
+ longitude: record.coordinates.last
95
+ }
96
+ }
97
+ end
98
+
99
+ def map_options
100
+ @resource.map_view || {}
101
+ end
102
+
103
+ def marker_proc
104
+ map_options[:record_marker] || default_record_marker_proc
105
+ end
106
+
107
+ def resource_mappable?
108
+ map_options[:record_marker].present? || map_options[:extra_markers].present? || @resources.first.record.respond_to?(:coordinates)
109
+ end
110
+ end