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,107 @@
1
+ .field-wrapper {
2
+ @apply relative flex flex-col grow leading-tight size-auto w-full md:flex-row md:items-center py-1 w-full grow-0 text-ellipsis overflow-auto;
3
+ }
4
+
5
+ .field-wrapper__label {
6
+ @apply flex self-start items-center shrink-0 w-full px-4 font-medium text-content text-sm md:w-52 min-h-10;
7
+ }
8
+
9
+ .field-wrapper__help {
10
+ @apply text-content-secondary mt-2 text-xs leading-4;
11
+ }
12
+
13
+ .field-wrapper__label-help {
14
+ @apply text-content-secondary text-xs leading-none font-normal;
15
+ }
16
+
17
+ .field-wrapper__error {
18
+ @apply text-danger-content mt-2 text-xs leading-4 inline-flex items-center gap-0.5;
19
+ }
20
+
21
+ .field-wrapper__error::before {
22
+ content: "";
23
+ @apply inline-block size-3 shrink-0;
24
+ background-color: currentColor;
25
+ -webkit-mask: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' stroke-width='2' stroke='currentColor' fill='none' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath stroke='none' d='M0 0h24v24H0z' fill='none'/%3E%3Ccircle cx='12' cy='12' r='9'/%3E%3Cpath d='M10 8l4 8'/%3E%3Cpath d='M10 16l4 -8'/%3E%3C/svg%3E") no-repeat center / contain;
26
+ mask: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' stroke-width='2' stroke='currentColor' fill='none' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath stroke='none' d='M0 0h24v24H0z' fill='none'/%3E%3Ccircle cx='12' cy='12' r='9'/%3E%3Cpath d='M10 8l4 8'/%3E%3Cpath d='M10 16l4 -8'/%3E%3C/svg%3E") no-repeat center / contain;
27
+ }
28
+
29
+ .field-wrapper__content {
30
+ @apply flex-1 flex flex-row px-4 py-1;
31
+ }
32
+
33
+ .field-wrapper__content-wrapper {
34
+ @apply relative w-full md:w-8/12;
35
+ }
36
+
37
+ .field-wrapper__input {
38
+ @apply relative w-full;
39
+ }
40
+
41
+ /* ==========================================================================
42
+ Loading Spinner
43
+ ========================================================================== */
44
+
45
+ .input-wrapper--loading {
46
+ @apply relative w-full;
47
+ }
48
+
49
+ .input-wrapper--loading::after {
50
+ @apply absolute top-1/2 rounded-full border-2 border-content border-e-tertiary pointer-events-none;
51
+
52
+ content: "";
53
+ inset-inline-end: var(--input-icon-offset);
54
+ width: var(--input-icon-size);
55
+ height: var(--input-icon-size);
56
+ transform: translateY(-50%);
57
+ transform-origin: center;
58
+ animation: input-spinner-rotation 0.9s linear infinite;
59
+ }
60
+
61
+ @keyframes input-spinner-rotation {
62
+ from {
63
+ transform: translateY(-50%) rotate(0deg);
64
+ }
65
+ to {
66
+ transform: translateY(-50%) rotate(360deg);
67
+ }
68
+ }
69
+
70
+ /* disabled & loading state */
71
+ .input-wrapper--loading:has(input:disabled)::after {
72
+ @apply border-content-secondary border-e-tertiary cursor-not-allowed;
73
+ }
74
+
75
+ .field-wrapper--full-width {
76
+ .field-wrapper__content-wrapper {
77
+ @apply w-full;
78
+ }
79
+ }
80
+
81
+ /* Stacked modifier */
82
+ .all-fields-stacked .field-wrapper,
83
+ .field-wrapper.field-wrapper--stacked {
84
+ @apply md:flex-col md:items-start gap-y-2 py-3;
85
+
86
+ .field-wrapper__label {
87
+ @apply md:w-full min-h-0 pt-0;
88
+ }
89
+
90
+ .field-wrapper__label-help {
91
+ @apply pb-0;
92
+ }
93
+
94
+ .field-wrapper__content {
95
+ @apply pt-0 px-4 w-full;
96
+ }
97
+
98
+ .field-wrapper__content-wrapper {
99
+ @apply w-full;
100
+ }
101
+ }
102
+
103
+ .container-small {
104
+ .field-wrapper__content-wrapper {
105
+ @apply w-full;
106
+ }
107
+ }
@@ -0,0 +1,61 @@
1
+ /* Grid Card Component - Based on Figma Design System */
2
+ /* BEM Methodology: .grid-card (block), .grid-card__element (elements) */
3
+
4
+ .grid-wrapper {
5
+ @apply w-full grid grid-cols-1 xs:grid-cols-2 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-3 xl:grid-cols-4 2xl:grid-cols-6 gap-4 mt-8;
6
+ }
7
+
8
+ .grid-card {
9
+ @apply flex flex-col flex-1 shrink-0 basis-0 items-start border rounded-xl p-2 gap-2 min-w-52 bg-primary border-tertiary;
10
+
11
+ box-shadow: var(--box-shadow-grid-card);
12
+ }
13
+
14
+ .grid-card__image {
15
+ @apply relative w-full overflow-hidden rounded-lg h-40 bg-secondary;
16
+ }
17
+
18
+ .grid-card input[type="checkbox"] {
19
+ @apply hidden;
20
+ }
21
+
22
+ .grid-card input[type="checkbox"]:checked,
23
+ .grid-card:hover input[type="checkbox"] {
24
+ @apply block z-10 absolute top-1 start-1;
25
+
26
+ /* Override mx-2 from item selector so checkbox aligns to card edge */
27
+ margin: 0 !important;
28
+ }
29
+
30
+ .grid-card__actions-row {
31
+ @apply flex items-center justify-end w-full gap-2 self-stretch;
32
+ }
33
+
34
+ .grid-card__badge {
35
+ @apply flex-1 shrink-0 basis-0 flex;
36
+ }
37
+
38
+ .grid-card__action {
39
+ @apply border-none bg-transparent p-1.5;
40
+ box-shadow: none;
41
+ }
42
+
43
+ .grid-card__action:hover {
44
+ @apply bg-transparent;
45
+ }
46
+
47
+ .grid-card__content {
48
+ @apply flex flex-col items-start w-full px-2 pb-2;
49
+ }
50
+
51
+ .grid-card__title {
52
+ @apply text-content text-base font-medium self-stretch leading-6;
53
+ }
54
+
55
+ .grid-card__title a {
56
+ color: inherit;
57
+ }
58
+
59
+ .grid-card__description {
60
+ @apply text-content-secondary text-xs font-normal self-stretch leading-4;
61
+ }
@@ -0,0 +1,312 @@
1
+ :root {
2
+ --input-py: --spacing(1);
3
+ --input-font-size: --spacing(3.5);
4
+ --input-leading: --spacing(5.5);
5
+ --input-border-w: 1px;
6
+
7
+ --input-icon-offset: --spacing(2);
8
+ --input-icon-size: --spacing(4.5);
9
+ --input-icon-gap: --spacing(1);
10
+ }
11
+
12
+ /* ==========================================================================
13
+ Base
14
+ ========================================================================== */
15
+ @layer base {
16
+ select,
17
+ textarea,
18
+ .textarea-field,
19
+ input:not([type="range"]):not([type="checkbox"]):not([type="radio"]),
20
+ .input-field,
21
+ tags.tagify {
22
+ @apply appearance-none inline-flex bg-primary text-content
23
+ rounded-lg border border-tertiary ps-2 pe-2 font-normal overflow-hidden
24
+ text-ellipsis whitespace-nowrap relative w-full;
25
+
26
+ /* box-shadow: var(--box-shadow); */
27
+ padding-block: var(--input-py);
28
+ font-size: var(--input-font-size);
29
+ line-height: var(--input-leading);
30
+ }
31
+
32
+ tags.tagify {
33
+ @apply overflow-visible whitespace-normal px-2 text-clip;
34
+ }
35
+
36
+ textarea,
37
+ .textarea-field {
38
+ @apply block text-ellipsis whitespace-normal wrap-break-word;
39
+ }
40
+
41
+ /* ==========================================================================
42
+ Size Variants — Tailwind utilities + token overrides
43
+ ========================================================================== */
44
+
45
+ .input--size-sm {
46
+ --input-py: --spacing(1);
47
+ --input-font-size: --spacing(3);
48
+ --input-leading: --spacing(4);
49
+
50
+ --input-icon-size: --spacing(4);
51
+ }
52
+
53
+ .input--size-md {
54
+ --input-py: --spacing(1);
55
+ --input-font-size: --spacing(3.5);
56
+ --input-leading: --spacing(5.5);
57
+
58
+ --input-icon-size: --spacing(4.5);
59
+ }
60
+
61
+ .input--size-lg {
62
+ --input-py: --spacing(2);
63
+ --input-font-size: --spacing(3.5);
64
+ --input-leading: --spacing(5.5);
65
+
66
+ --input-icon-size: --spacing(5);
67
+ }
68
+
69
+ /* For icons and with shortcut search command + k where the size depends on the parent size*/
70
+ :has(> .input--size-sm) {
71
+ --input-icon-size: --spacing(4);
72
+ --input-leading: --spacing(3.5);
73
+ }
74
+ :has(> .input--size-md) {
75
+ --input-icon-size: --spacing(4.5);
76
+ --input-leading: --spacing(4);
77
+ }
78
+ :has(> .input--size-lg) {
79
+ --input-icon-size: --spacing(5);
80
+ --input-leading: --spacing(4.5);
81
+ }
82
+
83
+ /* ==========================================================================
84
+ Disabled State
85
+ ========================================================================== */
86
+
87
+ /* Disabled input — cursor on parent (input has pointer-events-none) */
88
+ .field-wrapper__content-wrapper:has(input:disabled),
89
+ .field-wrapper__content-wrapper:has(select:disabled),
90
+ .field-wrapper__content-wrapper:has(textarea:disabled),
91
+ .field-wrapper__content-wrapper:has(.input-field:disabled) {
92
+ @apply cursor-not-allowed;
93
+ }
94
+
95
+ .field-wrapper[data-field-type="trix"] .field-wrapper__content-wrapper {
96
+ @apply cursor-auto;
97
+ }
98
+
99
+ select:disabled,
100
+ textarea:disabled,
101
+ .textarea-field:disabled,
102
+ input:not([type="range"]):not([type="checkbox"]):not([type="radio"]):disabled,
103
+ .input-field:disabled {
104
+ @apply pointer-events-none text-content-secondary;
105
+ }
106
+ /* ==========================================================================
107
+ Placeholder
108
+ ========================================================================== */
109
+
110
+ *::placeholder {
111
+ @apply text-content-secondary;
112
+ }
113
+
114
+ /* ==========================================================================
115
+ States: Focus
116
+ ========================================================================== */
117
+
118
+ select:focus-visible,
119
+ textarea:focus-visible,
120
+ .textarea-field:focus-visible,
121
+ input:not([type="range"]):not([type="checkbox"]):not([type="radio"]):focus-visible,
122
+ .input-field:focus-visible
123
+ {
124
+ @apply outline-none;
125
+ box-shadow: var(--box-shadow-focus);
126
+ }
127
+ }
128
+
129
+ @layer component {
130
+ /* ==========================================================================
131
+ States: Error, Success
132
+ ========================================================================== */
133
+ select.input-field--error,
134
+ textarea.input-field--error,
135
+ .textarea-field.input-field--error,
136
+ input:not([type="range"]):not([type="checkbox"]):not([type="radio"]).input-field--error,
137
+ .input-field.input-field--error
138
+ {
139
+ box-shadow: var(--box-shadow-error);
140
+ }
141
+
142
+ select.input-field--success,
143
+ textarea.input-field--success,
144
+ .textarea-field.input-field--success,
145
+ input:not([type="range"]):not([type="checkbox"]):not([type="radio"]).input-field--success,
146
+ .input-field.input-field--success {
147
+ box-shadow: var(--box-shadow-success);
148
+ }
149
+
150
+ /* ==========================================================================
151
+ Color Input
152
+ ========================================================================== */
153
+
154
+ input[type="color"] {
155
+ @apply cursor-pointer;
156
+
157
+ height: calc(var(--input-leading) + var(--input-py) * 2 + var(--input-border-w) * 2);
158
+ }
159
+
160
+ /* ==========================================================================
161
+ Password Visibility
162
+ ========================================================================== */
163
+
164
+ .input-field__password-toggle {
165
+ @apply absolute text-content-secondary inset-y-0 end-0 flex items-center cursor-pointer;
166
+
167
+ padding-inline-end: var(--input-icon-offset);
168
+ }
169
+
170
+ /* Wrapper-based selector: reliably targets password inputs with visibility toggle
171
+ (User create resource, password confirmation, etc.) */
172
+ .field-wrapper__input:has(.input-field__password-toggle) input {
173
+ padding-inline-end: calc(
174
+ var(--input-icon-size) +
175
+ var(--input-icon-offset) +
176
+ var(--input-icon-gap)
177
+ );
178
+ }
179
+
180
+ /* Unified icon sizing — reads from size token */
181
+ .input-field__password-toggle svg,
182
+ .input-wrapper--loading::after,
183
+ .search-input__prefix svg {
184
+ @apply shrink-0;
185
+ width: var(--input-icon-size);
186
+ height: var(--input-icon-size);
187
+ }
188
+
189
+ /* Loading: show spinner only — hide visibility toggle */
190
+ .input-wrapper--loading .input-field__password-toggle {
191
+ @apply hidden;
192
+ }
193
+
194
+ /* ==========================================================================
195
+ Select Input — Custom Chevron (background, light + dark mode)
196
+ ========================================================================== */
197
+
198
+ select:not([multiple]) {
199
+ @apply cursor-pointer text-content;
200
+ padding-inline-end: calc(var(--input-icon-offset) + var(--input-icon-size) + var(--input-icon-gap));
201
+ /* Light: text-content (--color-content) */
202
+ background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23050817' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath stroke='none' d='M0 0h24v24H0z' fill='none'/%3E%3Cpath d='M6 9l6 6l6 -6'/%3E%3C/svg%3E");
203
+ background-repeat: no-repeat;
204
+ background-position: right var(--input-icon-offset) center;
205
+ background-size: var(--input-icon-size);
206
+ }
207
+
208
+ .dark select:not([multiple]) {
209
+ /* Dark: text-content (--color-content = white) */
210
+ background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23ffffff' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath stroke='none' d='M0 0h24v24H0z' fill='none'/%3E%3Cpath d='M6 9l6 6l6 -6'/%3E%3C/svg%3E");
211
+ }
212
+
213
+ [dir="rtl"] select:not([multiple]) {
214
+ background-position: left var(--input-icon-offset) center;
215
+ }
216
+
217
+ select[multiple] {
218
+ background-image: none;
219
+ }
220
+
221
+ /* ==========================================================================
222
+ Date Input
223
+ ========================================================================== */
224
+
225
+ input[type="date"]::-webkit-calendar-picker-indicator,
226
+ .input-field--date::-webkit-calendar-picker-indicator {
227
+ @apply opacity-0 absolute end-0 top-0 h-full cursor-pointer;
228
+ }
229
+
230
+ input[type="date"],
231
+ .input-field--date {
232
+ @apply bg-no-repeat relative;
233
+ background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%236b7280' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath stroke='none' d='M0 0h24v24H0z' fill='none'/%3E%3Cpath d='M4 7a2 2 0 0 1 2 -2h12a2 2 0 0 1 2 2v12a2 2 0 0 1 -2 2h-12a2 2 0 0 1 -2 -2v-12'/%3E%3Cpath d='M16 3v4'/%3E%3Cpath d='M8 3v4'/%3E%3Cpath d='M4 11h16'/%3E%3Cpath d='M11 15h1'/%3E%3Cpath d='M12 15v3'/%3E%3C/svg%3E");
234
+ background-position: right var(--input-icon-offset) center;
235
+ background-size: var(--input-icon-size);
236
+ }
237
+
238
+ input[type="date"]:disabled,
239
+ .input-field--date:disabled {
240
+ background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%239ca3af' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath stroke='none' d='M0 0h24v24H0z' fill='none'/%3E%3Cpath d='M4 7a2 2 0 0 1 2 -2h12a2 2 0 0 1 2 2v12a2 2 0 0 1 -2 2h-12a2 2 0 0 1 -2 -2v-12'/%3E%3Cpath d='M16 3v4'/%3E%3Cpath d='M8 3v4'/%3E%3Cpath d='M4 11h16'/%3E%3Cpath d='M11 15h1'/%3E%3Cpath d='M12 15v3'/%3E%3C/svg%3E");
241
+ }
242
+
243
+ /* ==========================================================================
244
+ Search Input
245
+ ========================================================================== */
246
+
247
+ .search-input {
248
+ @apply relative w-full;
249
+ }
250
+
251
+ .search-input__prefix {
252
+ @apply absolute inset-y-0 start-0 flex items-center pointer-events-none text-content-secondary z-10 ps-[var(--input-icon-offset)];
253
+ }
254
+
255
+ /* Input padding — prefix side, suffix side (no shortcut / with shortcut) */
256
+ input[type=search],
257
+ .search-input__input {
258
+ padding-inline-start: calc(
259
+ var(--input-icon-offset) +
260
+ var(--input-icon-size) +
261
+ var(--input-icon-gap)
262
+ );
263
+ padding-inline-end: calc(
264
+ var(--input-icon-offset) +
265
+ var(--input-icon-gap)
266
+ );
267
+ }
268
+
269
+ /* Default (Mac): ⌘ + K — narrower */
270
+ .search-input__input.search-input__input--with-shortcut {
271
+ padding-inline-end: calc(
272
+ var(--input-icon-offset) +
273
+ var(--input-icon-size) * 2 +
274
+ var(--input-icon-gap) * 3
275
+ );
276
+ }
277
+
278
+ /* PC: CTRL + K — wider */
279
+ body.os-pc .search-input__input.search-input__input--with-shortcut {
280
+ padding-inline-end: calc(
281
+ var(--input-icon-offset) +
282
+ var(--input-icon-size) * 3 +
283
+ var(--input-icon-gap) * 4
284
+ );
285
+ }
286
+
287
+ /* Suffix (shortcut) */
288
+ .search-input__suffix {
289
+ @apply absolute inset-y-0 end-0 flex items-center text-content-secondary pointer-events-none z-10 gap-1;
290
+ padding-inline-end: var(--input-icon-offset);
291
+ }
292
+
293
+ .search-input__shortcut {
294
+ @apply text-xs font-semibold py-px px-1 border border-secondary rounded-sm bg-secondary min-w-5 flex items-center justify-center;
295
+
296
+ line-height: var(--input-leading);
297
+ box-shadow: var(--box-shadow-search-input-shortcut);
298
+ }
299
+
300
+ /* ==========================================================================
301
+ Range Input
302
+ ========================================================================== */
303
+
304
+ input[type="range"] {
305
+ @apply accent-accent ;
306
+ }
307
+
308
+ /* Thumb (WebKit + Firefox) */
309
+ input[type="range"]::-webkit-slider-thumb {
310
+ @apply accent-accent-content bg-accent-content;
311
+ }
312
+ }
@@ -0,0 +1,228 @@
1
+ /* Modal Component — composes .card for the content shell */
2
+
3
+ /* Height size variables */
4
+ .modal {
5
+ --modal-height-sm: 24rem;
6
+ --modal-height-md: 28rem;
7
+ --modal-height-lg: 32rem;
8
+ --modal-height-xl: 36rem;
9
+ --modal-height-2xl: 42rem;
10
+ --modal-height-3xl: 48rem;
11
+ --modal-height-4xl: 56rem;
12
+ }
13
+
14
+ .modal {
15
+ @apply fixed inset-0 w-full z-150 flex justify-center items-center;
16
+ }
17
+
18
+ .modal__overlay {
19
+ @apply absolute w-full h-full flex justify-center items-center;
20
+
21
+ background-color: color-mix(in oklch, var(--color-avo-neutral-950) 25%, transparent);
22
+
23
+ .dark & {
24
+ background-color: color-mix(in oklch, var(--color-avo-neutral-950) 60%, transparent);
25
+ }
26
+ }
27
+
28
+ .modal__card-container {
29
+ @apply flex flex-col items-center justify-center relative shrink-0 w-full;
30
+ }
31
+
32
+ .modal__card {
33
+ @apply relative size-full shrink-0;
34
+
35
+ box-shadow: var(--shadow-modal);
36
+ }
37
+
38
+ /* Modal-specific card overrides */
39
+ .modal__card > .card {
40
+ @apply size-full;
41
+ }
42
+
43
+ .modal__card > .card > .card__wrapper {
44
+ @apply size-full flex flex-col;
45
+ }
46
+
47
+ .modal__card .card__header {
48
+ @apply py-3 px-4;
49
+ }
50
+
51
+ .modal__card .card__body {
52
+ @apply flex flex-col flex-1 gap-4 items-start pt-0 px-0 min-h-64;
53
+ }
54
+
55
+ .modal__title {
56
+ @apply font-medium;
57
+ }
58
+
59
+ .modal__close-button {
60
+ @apply shrink-0 size-5 cursor-pointer text-content-secondary hover:text-content transition-colors;
61
+ }
62
+
63
+ .modal__width-selector {
64
+ @apply shrink-0 size-5 cursor-pointer text-content-secondary hover:text-content transition-colors;
65
+ }
66
+
67
+ .modal__height-selector {
68
+ @apply shrink-0 size-5 cursor-pointer text-content-secondary hover:text-content transition-colors;
69
+ }
70
+
71
+ .modal__body-content {
72
+ @apply size-full;
73
+ }
74
+
75
+ .modal__controls {
76
+ @apply flex items-center justify-between py-2 px-3 w-full;
77
+ }
78
+
79
+ .modal--width-sm {
80
+ .modal__card {
81
+ @apply w-sm;
82
+ }
83
+ }
84
+
85
+ .modal--width-md {
86
+ .modal__card {
87
+ @apply w-md;
88
+ }
89
+ }
90
+
91
+ .modal--width-lg {
92
+ .modal__card {
93
+ @apply w-lg;
94
+ }
95
+ }
96
+
97
+ .modal--width-xl {
98
+ .modal__card {
99
+ @apply w-xl;
100
+ }
101
+ }
102
+
103
+ .modal--width-2xl {
104
+ .modal__card {
105
+ @apply w-2xl;
106
+ }
107
+ }
108
+
109
+ .modal--width-3xl {
110
+ .modal__card {
111
+ @apply w-3xl;
112
+ }
113
+ }
114
+
115
+ .modal--width-4xl {
116
+ .modal__card {
117
+ @apply w-4xl;
118
+ }
119
+ }
120
+
121
+ .modal--width-25 {
122
+ .modal__card {
123
+ @apply w-[25dvw];
124
+ }
125
+ }
126
+
127
+ .modal--width-50 {
128
+ .modal__card {
129
+ @apply w-[50dvw];
130
+ }
131
+ }
132
+
133
+ .modal--width-75 {
134
+ .modal__card {
135
+ @apply w-[75dvw];
136
+ }
137
+ }
138
+
139
+ .modal--width-100 {
140
+ .modal__card {
141
+ @apply w-full;
142
+ }
143
+ }
144
+
145
+ .modal--width-full {
146
+ .modal__card {
147
+ @apply w-[calc(100dvw-2rem)];
148
+ }
149
+ }
150
+
151
+ .modal--height-auto {
152
+ .modal__card {
153
+ @apply min-h-32;
154
+ }
155
+ }
156
+
157
+ .modal--height-sm {
158
+ .modal__card {
159
+ @apply h-[var(--modal-height-sm)];
160
+ }
161
+ }
162
+
163
+ .modal--height-md {
164
+ .modal__card {
165
+ @apply h-[var(--modal-height-md)];
166
+ }
167
+ }
168
+
169
+ .modal--height-lg {
170
+ .modal__card {
171
+ @apply h-[var(--modal-height-lg)];
172
+ }
173
+ }
174
+
175
+ .modal--height-xl {
176
+ .modal__card {
177
+ @apply h-[var(--modal-height-xl)];
178
+ }
179
+ }
180
+
181
+ .modal--height-2xl {
182
+ .modal__card {
183
+ @apply h-[var(--modal-height-2xl)];
184
+ }
185
+ }
186
+
187
+ .modal--height-3xl {
188
+ .modal__card {
189
+ @apply h-[var(--modal-height-3xl)];
190
+ }
191
+ }
192
+
193
+ .modal--height-4xl {
194
+ .modal__card {
195
+ @apply h-[var(--modal-height-4xl)];
196
+ }
197
+ }
198
+
199
+ .modal--height-full {
200
+ .modal__card {
201
+ @apply h-[calc(100dvh-2rem)];
202
+ }
203
+ }
204
+
205
+
206
+ .modal--height-25 {
207
+ .modal__card {
208
+ @apply h-[25dvh];
209
+ }
210
+ }
211
+
212
+ .modal--height-50 {
213
+ .modal__card {
214
+ @apply h-[50dvh];
215
+ }
216
+ }
217
+
218
+ .modal--height-75 {
219
+ .modal__card {
220
+ @apply h-[75dvh];
221
+ }
222
+ }
223
+
224
+ .modal--height-100 {
225
+ .modal__card {
226
+ @apply h-full;
227
+ }
228
+ }