plutonium 0.10.3 → 0.11.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (185) hide show
  1. checksums.yaml +4 -4
  2. data/Rakefile +1 -2
  3. data/app/assets/application.js.bk +31419 -0
  4. data/app/assets/plutonium-logo-original.png +0 -0
  5. data/app/assets/plutonium-logo-white.png +0 -0
  6. data/app/assets/plutonium-logo.png +0 -0
  7. data/app/assets/plutonium.css +1 -0
  8. data/app/assets/plutonium.ico +0 -0
  9. data/app/assets/plutonium.js +12416 -0
  10. data/app/assets/plutonium.js.map +7 -0
  11. data/app/assets/plutonium.min.js +39 -0
  12. data/app/assets/plutonium.min.js.map +7 -0
  13. data/app/views/application/_flash_alerts.html.erb +2 -2
  14. data/app/views/application/_resource_header.html.erb +263 -697
  15. data/app/views/application/_resource_sidebar.html.erb +14 -12
  16. data/app/views/components/action_button/action_button_component.rb +3 -3
  17. data/app/views/components/attributes.rb +184 -0
  18. data/app/views/components/base.rb +19 -40
  19. data/app/views/components/block/block_component.html.erb +1 -1
  20. data/app/views/components/block/block_component.rb +11 -7
  21. data/app/views/components/breadcrumbs/breadcrumbs_component.rb +3 -3
  22. data/app/views/components/button/button_component.html.erb +2 -2
  23. data/app/views/components/button/button_component.rb +10 -5
  24. data/app/views/components/dyna_frame_content/dyna_frame_content_component.html.erb +1 -0
  25. data/app/views/components/dyna_frame_content/dyna_frame_content_component.rb +3 -3
  26. data/app/views/components/dyna_frame_host/dyna_frame_host_component.html.erb +1 -2
  27. data/app/views/components/dyna_frame_host/dyna_frame_host_component.rb +12 -5
  28. data/app/views/components/empty_card/empty_card_component.rb +3 -3
  29. data/app/views/components/form/form_builder.rb +1 -1
  30. data/app/views/components/form/form_component.rb +3 -3
  31. data/app/views/components/has_many_panel/has_many_panel_component.html.erb +25 -0
  32. data/app/views/components/has_many_panel/has_many_panel_component.rb +16 -0
  33. data/app/views/components/header/header_component.rb +3 -3
  34. data/app/views/components/interactive_action_form/interactive_action_form_component.rb +3 -3
  35. data/app/views/components/nav_grid_menu/nav_grid_menu_component.html.erb +24 -0
  36. data/app/views/components/nav_grid_menu/nav_grid_menu_component.rb +23 -0
  37. data/app/views/components/nav_grid_menu_item/nav_grid_menu_item_component.html.erb +4 -0
  38. data/app/views/components/nav_grid_menu_item/nav_grid_menu_item_component.rb +20 -0
  39. data/app/views/components/nav_user/nav_user_component.html.erb +50 -0
  40. data/app/views/components/nav_user/nav_user_component.rb +32 -0
  41. data/app/views/components/nav_user_link/nav_user_link_component.html.erb +7 -0
  42. data/app/views/components/nav_user_link/nav_user_link_component.rb +23 -0
  43. data/app/views/components/nav_user_section/nav_user_section_component.html.erb +7 -0
  44. data/app/views/components/nav_user_section/nav_user_section_component.rb +18 -0
  45. data/app/views/components/nested_resource_form_fields/nested_resource_form_fields_component.html.erb +1 -3
  46. data/app/views/components/nested_resource_form_fields/nested_resource_form_fields_component.rb +11 -5
  47. data/app/views/components/pagination/pagination_component.html.erb +1 -1
  48. data/app/views/components/pagination/pagination_component.rb +10 -7
  49. data/app/views/components/panel/panel_component.html.erb +13 -6
  50. data/app/views/components/panel/panel_component.rb +11 -5
  51. data/app/views/components/resource_header/resource_header_component.html.erb +81 -0
  52. data/app/views/components/resource_header/resource_header_component.rb +20 -0
  53. data/app/views/components/resource_layout/resource_layout_component.html.erb +32 -0
  54. data/app/views/components/resource_layout/resource_layout_component.rb +39 -0
  55. data/app/views/components/sidebar/sidebar_component.html.erb +3 -33
  56. data/app/views/components/sidebar/sidebar_component.rb +3 -3
  57. data/app/views/components/sidebar_menu/sidebar_menu_component.html.erb +4 -2
  58. data/app/views/components/sidebar_menu/sidebar_menu_component.rb +10 -6
  59. data/app/views/components/sidebar_menu_item/sidebar_menu_item_component.html.erb +63 -71
  60. data/app/views/components/sidebar_menu_item/sidebar_menu_item_component.rb +27 -8
  61. data/app/views/components/skeleton/table/table_component.html.erb +1 -1
  62. data/app/views/components/skeleton/table/table_component.rb +3 -3
  63. data/app/views/components/table/table_component.html.erb +40 -89
  64. data/app/views/components/table/table_component.rb +124 -28
  65. data/app/views/components/table_search_input/table_search_input_component.html.erb +1 -1
  66. data/app/views/components/table_search_input/table_search_input_component.rb +11 -6
  67. data/app/views/components/table_toolbar/table_toolbar_component.html.erb +1 -1
  68. data/app/views/components/table_toolbar/table_toolbar_component.rb +11 -3
  69. data/app/views/components/toolbar/toolbar_component.html.erb +2 -2
  70. data/app/views/components/toolbar/toolbar_component.rb +16 -8
  71. data/app/views/layouts/resource.html.erb +21 -37
  72. data/app/views/layouts/rodauth.html.erb +32 -30
  73. data/app/views/resource/_interactive_resource_action_form.html.erb +1 -1
  74. data/app/views/resource/_resource_details.html.erb +8 -5
  75. data/app/views/resource/_resource_table.html.erb +70 -1
  76. data/app/views/resource/interactive_resource_collection_action.html.erb +1 -0
  77. data/app/views/resource/interactive_resource_record_action.html.erb +1 -0
  78. data/app/views/resource/interactive_resource_recordless_action.html.erb +1 -0
  79. data/app/views/resource/new.html.erb +1 -0
  80. data/config/initializers/simple_form.rb +22 -2
  81. data/esbuild.config.js +35 -31
  82. data/lib/generators/pu/core/assets/assets_generator.rb +41 -0
  83. data/lib/generators/pu/core/assets/templates/tailwind.config.js +18 -0
  84. data/lib/generators/pu/core/install/templates/config/initializers/plutonium.rb +3 -0
  85. data/lib/generators/pu/gem/dotenv/templates/config/initializers/001_ensure_required_env.rb +6 -0
  86. data/lib/generators/pu/gen/component/component_generator.rb +13 -10
  87. data/lib/generators/pu/gen/component/templates/component.html.erb.tt +1 -1
  88. data/lib/generators/pu/gen/component/templates/component.rb.tt +10 -4
  89. data/lib/generators/pu/pkg/app/app_generator.rb +4 -4
  90. data/lib/generators/pu/pkg/app/templates/app/controllers/concerns/controller.rb.tt +28 -0
  91. data/lib/generators/pu/pkg/app/templates/app/controllers/controller.rb.tt +5 -0
  92. data/lib/generators/pu/pkg/app/templates/app/controllers/dashboard_controller.rb.tt +1 -1
  93. data/lib/generators/pu/res/conn/conn_generator.rb +4 -4
  94. data/lib/generators/pu/res/conn/templates/app/controllers/resource_controller.rb.tt +1 -1
  95. data/lib/generators/pu/res/model/model_generator.rb +3 -3
  96. data/lib/generators/pu/res/scaffold/templates/policy.rb.tt +6 -0
  97. data/lib/generators/pu/service/sidekiq/sidekiq_generator.rb +0 -5
  98. data/lib/generators/pu/service/sidekiq/templates/app/sidekiq/sidekiq_job.rb +0 -2
  99. data/lib/plutonium/config.rb +2 -14
  100. data/lib/plutonium/core/associations/renderers/basic_renderer.rb +28 -0
  101. data/lib/plutonium/core/associations/renderers/factory.rb +36 -0
  102. data/lib/plutonium/core/associations/renderers/has_many_renderer.rb +16 -0
  103. data/lib/plutonium/core/autodiscovery/association_renderer_discoverer.rb +31 -0
  104. data/lib/plutonium/core/controllers/authorizable.rb +13 -17
  105. data/lib/plutonium/core/controllers/base.rb +3 -7
  106. data/lib/plutonium/core/controllers/presentable.rb +6 -1
  107. data/lib/plutonium/core/definers/association_renderer_definer.rb +33 -0
  108. data/lib/plutonium/core/fields/inputs/checkbox_input.rb +13 -0
  109. data/lib/plutonium/core/fields/inputs/factory.rb +1 -0
  110. data/lib/plutonium/core/fields/inputs/polymorphic_belongs_to_association_input.rb +1 -1
  111. data/lib/plutonium/core/ui/detail.rb +1 -0
  112. data/lib/plutonium/helpers/application_helper.rb +8 -9
  113. data/lib/plutonium/helpers/assets_helper.rb +33 -0
  114. data/lib/plutonium/helpers/display_helper.rb +13 -0
  115. data/lib/plutonium/helpers/form_helper.rb +1 -1
  116. data/lib/plutonium/helpers.rb +1 -0
  117. data/lib/plutonium/icons.rb +12 -5
  118. data/lib/plutonium/pkg/app.rb +10 -0
  119. data/lib/plutonium/pundit/context.rb +18 -0
  120. data/lib/plutonium/pundit/policy_finder.rb +25 -0
  121. data/lib/plutonium/railtie.rb +20 -8
  122. data/lib/plutonium/reloader.rb +18 -7
  123. data/lib/plutonium/resource/controller.rb +4 -0
  124. data/lib/plutonium/resource/policy.rb +69 -47
  125. data/lib/plutonium/resource/presenter.rb +1 -0
  126. data/lib/plutonium/resource/query_object.rb +139 -130
  127. data/lib/plutonium/rodauth/controller_methods.rb +7 -3
  128. data/lib/plutonium/version.rb +1 -1
  129. data/lib/plutonium.rb +9 -57
  130. data/package-lock.json +782 -17
  131. data/package.json +31 -8
  132. data/postcss.config.js +17 -7
  133. data/src/.npmignore +2 -0
  134. data/src/js/controllers/color_mode_controller.js +41 -0
  135. data/src/js/controllers/frame_navigator_controller.js +99 -0
  136. data/src/js/controllers/has_many_panel_controller.js +8 -0
  137. data/src/js/controllers/nav_grid_menu_controller.js +8 -0
  138. data/src/js/controllers/nav_grid_menu_item_controller.js +8 -0
  139. data/{app/views/components/tab_bar/tab_bar_controller.js → src/js/controllers/nav_user_controller.js} +2 -2
  140. data/src/js/controllers/nav_user_link_controller.js +8 -0
  141. data/src/js/controllers/nav_user_section_controller.js +8 -0
  142. data/src/js/controllers/register_controllers.js +45 -0
  143. data/{app/assets/javascripts → src/js}/controllers/resource_dismiss_controller.js +2 -0
  144. data/{app/assets/javascripts → src/js}/controllers/resource_drop_down_controller.js +2 -0
  145. data/src/js/controllers/resource_header_controller.js +8 -0
  146. data/src/js/controllers/resource_layout_controller.js +8 -0
  147. data/src/js/controllers/sidebar_menu_controller.js +8 -0
  148. data/src/js/controllers/sidebar_menu_item_controller.js +8 -0
  149. data/src/js/core.js +4 -0
  150. data/{app/assets/javascripts/plutonium-app.js → src/js/plutonium.js} +1 -1
  151. data/{app/assets/javascripts → src/js}/turbo/turbo_debug.js +2 -4
  152. data/tailwind.config.js +85 -84
  153. metadata +73 -39
  154. data/app/assets/build/plutonium.js +0 -5122
  155. data/app/assets/javascripts/controllers/index.js +0 -34
  156. data/app/assets/javascripts/plutonium.js +0 -1
  157. data/app/views/application/_color_modes.html.erb +0 -57
  158. data/app/views/components/tab_bar/tab_bar_component.html.erb +0 -11
  159. data/app/views/components/tab_bar/tab_bar_component.rb +0 -9
  160. data/app/views/resource/_nav_user.html.erb +0 -4
  161. data/app/views/resource/_tab_menu.html.erb +0 -13
  162. data/css.manifest +0 -3
  163. data/js.manifest +0 -4
  164. data/lib/generators/pu/pkg/app/templates/app/controllers/app_controller.rb.tt +0 -5
  165. data/lib/generators/pu/pkg/app/templates/app/controllers/package_controller.rb.tt +0 -26
  166. data/public/plutonium-assets/application.css +0 -25086
  167. data/public/plutonium-assets/plutonium-app-36KN5FVJ.js +0 -6
  168. data/public/plutonium-assets/plutonium-app-36KN5FVJ.js.map +0 -7
  169. data/public/plutonium-assets/plutonium-app-6WILQCTT.js +0 -39
  170. data/public/plutonium-assets/plutonium-app-6WILQCTT.js.map +0 -7
  171. data/public/plutonium-assets/plutonium.2d4f0c333cd000051d3b.css +0 -3424
  172. data/public/plutonium-assets/plutonium.50232e35b5495f5ad90d.css +0 -3415
  173. data/public/plutonium-assets/plutonium.8bee7a8482988b0360e3.css +0 -3420
  174. /data/{app/assets/build → lib/generators/pu/core/assets/templates}/.keep +0 -0
  175. /data/{app/assets/stylesheets → src/css}/plutonium.css +0 -0
  176. /data/{app/views/components/form → src/js/controllers}/form_controller.js +0 -0
  177. /data/{app/views/components/interactive_action_form → src/js/controllers}/interactive_action_form_controller.js +0 -0
  178. /data/{app/views/components/nested_resource_form_fields → src/js/controllers}/nested_resource_form_fields_controller.js +0 -0
  179. /data/{app/views/components/table → src/js/controllers}/table_controller.js +0 -0
  180. /data/{app/views/components/table_search_input → src/js/controllers}/table_search_input_controller.js +0 -0
  181. /data/{app/views/components/table_toolbar → src/js/controllers}/table_toolbar_controller.js +0 -0
  182. /data/{app/views/components/toolbar → src/js/controllers}/toolbar_controller.js +0 -0
  183. /data/{app/assets/javascripts → src/js}/turbo/index.js +0 -0
  184. /data/{app/assets/javascripts → src/js}/turbo/turbo_actions.js +0 -0
  185. /data/{app/assets/javascripts → src/js}/turbo/turbo_frame_monkey_patch.js +0 -0
@@ -1,61 +1,57 @@
1
- # TODO: make standard query type names e.g. search and scope configurable
2
-
3
1
  module Plutonium
4
2
  module Resource
5
- # The QueryObject class is responsible for handling various query types and applying them to the given scope.
6
3
  class QueryObject
7
4
  class << self
8
5
  end
9
6
 
10
- # The Query class serves as a base for different types of queries.
11
7
  class Query
12
8
  include Plutonium::Core::Definers::InputDefiner
13
9
 
14
- # Applies the query to the given scope with the provided parameters.
10
+ # Applies the query to the given scope using the provided parameters.
15
11
  #
16
- # @param scope [Object] the scope to apply the query on
17
- # @param params [Hash] the parameters for the query
18
- # @return [Object] the modified scope
12
+ # @param scope [Object] The initial scope to which the query will be applied.
13
+ # @param params [Hash] The parameters for the query.
14
+ # @return [Object] The modified scope.
19
15
  def apply(scope, params)
20
16
  params = extract_query_params(params)
21
- (input_definitions.size == params.size) ? apply_internal(scope, params) : scope
17
+
18
+ if input_definitions.size == params.size
19
+ apply_internal(scope, params)
20
+ else
21
+ scope
22
+ end
22
23
  end
23
24
 
24
25
  private
25
26
 
26
- # Raises an error, should be implemented by subclasses.
27
+ # Abstract method to apply the query logic to the scope.
28
+ # Should be implemented by subclasses.
27
29
  #
28
- # @param scope [Object] the scope to apply the query on
29
- # @param params [Hash] the parameters for the query
30
- # @raise [NotImplementedError] if not implemented by subclass
30
+ # @param scope [Object] The initial scope.
31
+ # @param params [Hash] The parameters for the query.
32
+ # @raise [NotImplementedError] If the method is not implemented.
31
33
  def apply_internal(scope, params)
32
34
  raise NotImplementedError, "#{self.class}#apply_internal"
33
35
  end
34
36
 
35
- # Extracts and processes query parameters.
37
+ # Extracts query parameters based on the defined inputs.
36
38
  #
37
- # @param params [Hash] the parameters to process
38
- # @return [Hash] the processed parameters
39
+ # @param params [Hash] The parameters to extract.
40
+ # @return [Hash] The extracted and symbolized parameters.
39
41
  def extract_query_params(params)
40
42
  input_definitions.collect_all(params).compact.symbolize_keys
41
43
  end
42
44
 
43
- # Returns the resource class, to be implemented by subclasses.
44
- #
45
- # @return [Class, nil] the resource class
46
- def resource_class
47
- nil
48
- end
45
+ # @return [nil] The resource class (default implementation returns nil).
46
+ def resource_class = nil
49
47
  end
50
48
 
51
- # The ScopeQuery class represents a query based on a scope.
52
49
  class ScopeQuery < Query
53
50
  attr_reader :name
54
51
 
55
- # Initializes a ScopeQuery.
52
+ # Initializes a ScopeQuery with a given name.
56
53
  #
57
- # @param name [Symbol] the name of the scope
58
- # @yield [self] optional block to configure the query
54
+ # @param name [Symbol] The name of the scope.
59
55
  def initialize(name)
60
56
  @name = name
61
57
  yield self if block_given?
@@ -63,24 +59,22 @@ module Plutonium
63
59
 
64
60
  private
65
61
 
66
- # Applies the scope query to the given scope with the provided parameters.
62
+ # Applies the scope query to the given scope.
67
63
  #
68
- # @param scope [Object] the scope to apply the query on
69
- # @param params [Hash] the parameters for the query
70
- # @return [Object] the modified scope
64
+ # @param scope [Object] The initial scope.
65
+ # @param params [Hash] The parameters for the query.
66
+ # @return [Object] The modified scope.
71
67
  def apply_internal(scope, params)
72
68
  scope.send(name, **params)
73
69
  end
74
70
  end
75
71
 
76
- # The BlockQuery class represents a query based on a block.
77
72
  class BlockQuery < Query
78
73
  attr_reader :body
79
74
 
80
- # Initializes a BlockQuery.
75
+ # Initializes a BlockQuery with a given block of code.
81
76
  #
82
- # @param body [Proc] the block to apply
83
- # @yield [self] optional block to configure the query
77
+ # @param body [Proc] The block of code for the query.
84
78
  def initialize(body)
85
79
  @body = body
86
80
  yield self if block_given?
@@ -88,22 +82,26 @@ module Plutonium
88
82
 
89
83
  private
90
84
 
91
- # Applies the block query to the given scope with the provided parameters.
85
+ # Applies the block query to the given scope.
92
86
  #
93
- # @param scope [Object] the scope to apply the query on
94
- # @param params [Hash] the parameters for the query
95
- # @return [Object] the modified scope
87
+ # @param scope [Object] The initial scope.
88
+ # @param params [Hash] The parameters for the query.
89
+ # @return [Object] The modified scope.
96
90
  def apply_internal(scope, params)
97
- (body.arity == 1) ? body.call(scope) : body.call(scope, **params)
91
+ if body.arity == 1
92
+ body.call(scope)
93
+ else
94
+ body.call(scope, **params)
95
+ end
98
96
  end
99
97
  end
100
98
 
101
- attr_reader :search_filter, :search_query, :context, :selected_sort_fields, :selected_sort_directions, :selected_scope_filter
99
+ attr_reader :search_filter, :search_query
102
100
 
103
- # Initializes a QueryObject.
101
+ # Initializes a QueryObject with the given context and parameters.
104
102
  #
105
- # @param context [Object] the context in which the queries are defined
106
- # @param params [Hash] the initial parameters for the query object
103
+ # @param context [Object] The context in which the query object is used.
104
+ # @param params [Hash] The parameters for initialization.
107
105
  def initialize(context, params)
108
106
  @context = context
109
107
 
@@ -116,59 +114,43 @@ module Plutonium
116
114
  extract_sort_params(params)
117
115
  end
118
116
 
119
- # Builds a URL with the current query parameters.
117
+ # Builds a URL with the given options for search and sorting.
120
118
  #
121
- # @param options [Hash] additional options for building the URL
122
- # @return [String] the built URL
119
+ # @param options [Hash] The options for building the URL.
120
+ # @return [String] The constructed URL with query parameters.
123
121
  def build_url(**options)
124
122
  q = {}
125
- q[:search] = options.fetch(:search, search_query).presence
126
- q[:scope] = options.fetch(:scope, selected_scope_filter).presence
123
+
124
+ q[:search] = options.key?(:search) ? options[:search].presence : search_query
125
+ q[:scope] = options.key?(:scope) ? options[:scope].presence : selected_scope_filter
126
+
127
127
  q[:sort_directions] = selected_sort_directions.dup
128
128
  q[:sort_fields] = selected_sort_fields.dup
129
-
130
- if (sort = options[:sort])
131
- handle_sort_options(q, sort, options[:reset])
132
- end
129
+ handle_sort_options!(q, options)
133
130
 
134
131
  "?#{{q: q}.to_param}"
135
132
  end
136
133
 
137
- # Applies the queries to the given scope.
134
+ # Applies the defined filters and sorts to the given scope.
138
135
  #
139
- # @param scope [Object] the scope to apply the queries on
140
- # @return [Object] the modified scope
136
+ # @param scope [Object] The initial scope to which filters and sorts are applied.
137
+ # @return [Object] The modified scope.
141
138
  def apply(scope)
142
139
  scope = search_filter.apply(scope, {search: search_query}) if search_filter.present?
143
140
  scope = scope_definitions[selected_scope_filter].apply(scope, {}) if selected_scope_filter.present?
144
- apply_sorters(scope)
141
+ apply_sorts(scope)
145
142
  end
146
143
 
147
- # Retrieves the scope definitions.
148
- #
149
- # @return [HashWithIndifferentAccess] the scope definitions
150
- def scope_definitions
151
- @scope_definitions ||= {}.with_indifferent_access
152
- end
144
+ def scope_definitions = @scope_definitions ||= {}.with_indifferent_access
153
145
 
154
- # Retrieves the filter definitions.
155
- #
156
- # @return [HashWithIndifferentAccess] the filter definitions
157
- def filter_definitions
158
- @filter_definitions ||= {}.with_indifferent_access
159
- end
146
+ def filter_definitions = @filter_definitions ||= {}.with_indifferent_access
160
147
 
161
- # Retrieves the sort definitions.
162
- #
163
- # @return [HashWithIndifferentAccess] the sort definitions
164
- def sort_definitions
165
- @sort_definitions ||= {}.with_indifferent_access
166
- end
148
+ def sort_definitions = @sort_definitions ||= {}.with_indifferent_access
167
149
 
168
- # Retrieves the sort parameters for a given name.
150
+ # Provides sorting parameters for the given field name.
169
151
  #
170
- # @param name [Symbol] the name of the sort field
171
- # @return [Hash, nil] the sort parameters or nil if not defined
152
+ # @param name [Symbol, String] The name of the field to sort.
153
+ # @return [Hash, nil] The sorting parameters including URL and direction.
172
154
  def sort_params_for(name)
173
155
  return unless sort_definitions[name]
174
156
 
@@ -182,59 +164,64 @@ module Plutonium
182
164
 
183
165
  private
184
166
 
185
- # Placeholder method for defining filters.
167
+ attr_reader :context, :selected_sort_fields, :selected_sort_directions, :selected_scope_filter
168
+
169
+ # Defines standard filters.
186
170
  def define_filters
171
+ # Implement filter definitions if needed
187
172
  end
188
173
 
189
- # Placeholder method for defining scopes.
174
+ # Defines standard scopes.
190
175
  def define_scopes
176
+ # Implement scope definitions if needed
191
177
  end
192
178
 
193
- # Placeholder method for defining sorters.
179
+ # Defines standard sorters.
194
180
  def define_sorters
181
+ # Implement sorter definitions if needed
195
182
  end
196
183
 
197
- # Defines standard queries.
184
+ # Defines standard queries for search and scope.
198
185
  def define_standard_queries
199
186
  define_search(:search) if resource_class.respond_to?(:search)
200
187
  end
201
188
 
202
- # Defines a filter.
189
+ # Defines a filter with the given name and body.
203
190
  #
204
- # @param name [Symbol] the name of the filter
205
- # @param body [Proc, nil] the body of the filter
206
- # @yield [Query] optional block to configure the query
191
+ # @param name [Symbol] The name of the filter.
192
+ # @param body [Proc, nil] The body of the filter.
207
193
  def define_filter(name, body = nil, &block)
208
194
  body ||= name
209
195
  filter_definitions[name] = build_query(body, &block)
210
196
  end
211
197
 
212
- # Defines a scope.
198
+ # Defines a scope with the given name and body.
213
199
  #
214
- # @param name [Symbol] the name of the scope
215
- # @param body [Proc, nil] the body of the scope
200
+ # @param name [Symbol] The name of the scope.
201
+ # @param body [Proc, nil] The body of the scope.
216
202
  def define_scope(name, body = nil)
217
203
  body ||= name
218
204
  scope_definitions[name] = build_query(body)
219
205
  end
220
206
 
221
- # Defines a sort.
207
+ # Defines a sort with the given name and body.
222
208
  #
223
- # @param name [Symbol] the name of the sort field
224
- # @param body [Proc, nil] the body of the sort
225
- def define_sort(name, body = nil)
209
+ # @param name [Symbol] The name of the sort.
210
+ # @param body [Proc, nil] The body of the sort.
211
+ def define_sorter(name, body = nil)
226
212
  if body.nil?
227
213
  sort_field = determine_sort_field(name)
228
214
  body = ->(scope, direction:) { scope.order(sort_field => direction) }
229
215
  end
216
+
230
217
  sort_definitions[name] = build_query(body) do |query|
231
218
  query.define_input :direction
232
219
  end
233
220
  end
234
221
 
235
- # Defines a search filter.
222
+ # Defines a search filter with the given body.
236
223
  #
237
- # @param body [Proc] the body of the search filter
224
+ # @param body [Proc, Symbol] The body of the search filter.
238
225
  def define_search(body)
239
226
  @search_filter = build_query(body) do |query|
240
227
  query.define_input :search
@@ -243,7 +230,7 @@ module Plutonium
243
230
 
244
231
  # Extracts filter parameters from the given params.
245
232
  #
246
- # @param params [Hash] the parameters to extract from
233
+ # @param params [Hash] The parameters to extract.
247
234
  def extract_filter_params(params)
248
235
  @search_query = params[:search]
249
236
  @selected_scope_filter = params[:scope]
@@ -251,39 +238,35 @@ module Plutonium
251
238
 
252
239
  # Extracts sort parameters from the given params.
253
240
  #
254
- # @param params [Hash] the parameters to extract from
241
+ # @param params [Hash] The parameters to extract.
255
242
  def extract_sort_params(params)
256
- @selected_sort_fields = Array(params[:sort_fields]) & sort_definitions.keys
257
- @selected_sort_directions = (params[:sort_directions]&.slice(*sort_definitions.keys) || {}).transform_values { |v| (v.upcase == "DESC") ? "DESC" : "ASC" }.with_indifferent_access
243
+ @selected_sort_fields = Array(params[:sort_fields])
244
+ @selected_sort_fields &= sort_definitions.keys
245
+
246
+ @selected_sort_directions = extract_sort_directions(params)
258
247
  end
259
248
 
260
- # Builds a query object.
249
+ # Builds a query object based on the given body and optional block.
261
250
  #
262
- # @param body [Symbol, Proc] the body of the query
263
- # @yield [Query] optional block to configure the query
264
- # @return [Query] the built query object
251
+ # @param body [Proc, Symbol] The body of the query.
252
+ # @yieldparam query [Query] The query object.
253
+ # @return [Query] The constructed query object.
265
254
  def build_query(body, &block)
266
255
  case body
267
256
  when Symbol
268
257
  raise "Cannot find scope :#{body} on #{resource_class}" unless resource_class.respond_to?(body)
258
+
269
259
  ScopeQuery.new(body, &block)
270
260
  else
271
261
  BlockQuery.new(body, &block)
272
262
  end
273
263
  end
274
264
 
275
- # Retrieves the resource class.
265
+ # Determines the sort field for the given name.
276
266
  #
277
- # @return [Class] the resource class
278
- def resource_class
279
- context.resource_class
280
- end
281
-
282
- # Determines the sort field for a given name.
283
- #
284
- # @param name [Symbol] the name of the sort field
285
- # @return [Symbol] the determined sort field
286
- # @raise [RuntimeError] if unable to determine sort logic for the field
267
+ # @param name [Symbol, String] The name of the field.
268
+ # @return [Symbol] The sort field.
269
+ # @raise [RuntimeError] If unable to determine sort logic.
287
270
  def determine_sort_field(name)
288
271
  if resource_class.primary_key == name.to_s || resource_class.content_column_field_names.include?(name)
289
272
  name
@@ -294,39 +277,65 @@ module Plutonium
294
277
  end
295
278
  end
296
279
 
297
- # Handles sorting options.
280
+ # Extracts sort directions from the given params.
281
+ #
282
+ # @param params [Hash] The parameters to extract.
283
+ # @return [Hash] The extracted sort directions.
284
+ def extract_sort_directions(params)
285
+ params[:sort_directions]&.slice(*sort_definitions.keys) || {}
286
+ end
287
+
288
+ # Handles the sort options for building the URL.
289
+ #
290
+ # @param query_params [Hash] The query parameters.
291
+ # @param options [Hash] The options for sorting.
292
+ def handle_sort_options!(query_params, options)
293
+ if (sort = options[:sort])
294
+ handle_sort_reset!(query_params, sort, options[:reset])
295
+ end
296
+ end
297
+
298
+ # Handles the reset option for sorting.
298
299
  #
299
- # @param query [Hash] the query parameters
300
- # @param sort [Symbol] the sort field
301
- # @param reset [Boolean] whether to reset the sorting
302
- def handle_sort_options(query, sort, reset)
300
+ # @param query_params [Hash] The query parameters.
301
+ # @param sort [Symbol, String] The sort field.
302
+ # @param reset [Boolean] Whether to reset the sort.
303
+ def handle_sort_reset!(query_params, sort, reset)
303
304
  if reset
304
- query[:sort_fields].delete_if { |e| e == sort.to_s }
305
- query[:sort_directions].delete(sort)
305
+ query_params[:sort_fields].delete_if { |e| e == sort.to_s }
306
+ query_params[:sort_directions].delete(sort)
306
307
  else
307
- query[:sort_fields] << sort.to_s unless query[:sort_fields].include?(sort.to_s)
308
+ query_params[:sort_fields] << sort.to_s unless query_params[:sort_fields].include?(sort.to_s)
309
+
308
310
  sort_direction = selected_sort_directions[sort]
309
- query[:sort_directions][sort] = if sort_direction.nil?
310
- "ASC"
311
+ if sort_direction.nil?
312
+ query_params[:sort_directions][sort] = "ASC"
313
+ elsif sort_direction == "ASC"
314
+ query_params[:sort_directions][sort] = "DESC"
311
315
  else
312
- ((sort_direction == "ASC") ? "DESC" : "ASC")
316
+ query_params[:sort_fields].delete_if { |e| e == sort.to_s }
317
+ query_params[:sort_directions].delete(sort)
313
318
  end
314
- query[:sort_fields].delete_if { |e| e == sort.to_s } if query[:sort_directions][sort] == "ASC"
315
319
  end
316
320
  end
317
321
 
318
- # Applies sorters to the scope.
322
+ # Applies the defined sorters to the given scope.
319
323
  #
320
- # @param scope [Object] the scope to apply sorters on
321
- # @return [Object] the modified scope
322
- def apply_sorters(scope)
324
+ # @param scope [Object] The initial scope.
325
+ # @return [Object] The modified scope.
326
+ def apply_sorts(scope)
323
327
  selected_sort_fields.each do |name|
324
328
  sorter = sort_definitions[name]
325
329
  next unless sorter.present?
326
- scope = sorter.apply(scope, {direction: selected_sort_directions[name] || "ASC"})
330
+
331
+ params = {direction: selected_sort_directions[name] || "ASC"}
332
+ scope = sorter.apply(scope, params)
327
333
  end
328
334
  scope
329
335
  end
336
+
337
+ # @return [Object] The resource class from the context.
338
+ def resource_class = context.resource_class
330
339
  end
331
340
  end
332
341
  end
@@ -4,15 +4,19 @@ module Plutonium
4
4
  extend ActiveSupport::Concern
5
5
 
6
6
  included do
7
+ helper Plutonium::Helpers::ApplicationHelper
8
+ helper Plutonium::Helpers::ComponentHelper
9
+ helper Plutonium::Helpers::AssetsHelper
10
+
7
11
  layout "rodauth"
8
12
  append_view_path File.expand_path("app/views", Plutonium.root)
9
- helper_method :application_name
13
+ helper_method :root_path
10
14
  end
11
15
 
12
16
  private
13
17
 
14
- def application_name
15
- Plutonium.application_name
18
+ def root_path
19
+ "/"
16
20
  end
17
21
  end
18
22
  end
@@ -1,3 +1,3 @@
1
1
  module Plutonium
2
- VERSION = "0.10.3"
2
+ VERSION = "0.11.0"
3
3
  end
data/lib/plutonium.rb CHANGED
@@ -24,70 +24,22 @@ module Plutonium
24
24
  end
25
25
 
26
26
  def self.application_name
27
- Rails.application.class.module_parent.name
27
+ @application_name || Rails.application.class.module_parent.name
28
28
  end
29
29
 
30
- def self.development?
31
- ActiveModel::Type::Boolean.new.cast(ENV["PLUTONIUM_DEV"]).present?
30
+ def self.application_name=(application_name)
31
+ @application_name = application_name
32
32
  end
33
33
 
34
- def self.stylesheet_link
35
- return @stylesheet_link if defined?(@stylesheet_link) && !development?
36
-
37
- if development?
38
- base_dir = "/plutonium-assets/build"
39
- filename = "plutonium-dev.css"
40
- else
41
- base_dir = "/plutonium-assets"
42
- filename = "plutonium.css"
43
- end
44
-
45
- file = stylesheet_manifest[filename]
46
- @stylesheet_link = "#{base_dir}/#{file}"
47
- end
48
-
49
- def self.script_link
50
- return @script_link if defined?(@script_link) && !development?
51
-
52
- filename = "plutonium-app.js"
53
- base_dir = if development?
54
- "/plutonium-assets/build"
55
- else
56
- "/plutonium-assets"
57
- end
58
-
59
- file = script_manifest[filename]
60
- @script_link = "#{base_dir}/#{file}"
61
- end
62
-
63
- def self.favicon_link
64
- "/plutonium-assets/plutonium.ico"
65
- end
66
-
67
- def self.logo_link
68
- "/plutonium-assets/plutonium-logo.png"
69
- end
70
-
71
- def self.stylesheet_manifest
72
- return @stylesheet_manifest if defined?(@stylesheet_manifest) && !development?
73
-
74
- manifest = if development?
75
- "css.dev.manifest"
76
- else
77
- "css.manifest"
78
- end
79
- @stylesheet_manifest = JSON.parse(File.read(root.join(manifest)))
34
+ def self.development?
35
+ ActiveModel::Type::Boolean.new.cast(ENV["PLUTONIUM_DEV"]).present?
80
36
  end
81
37
 
82
- def self.script_manifest
83
- return @script_manifest if defined?(@script_manifest) && !development?
38
+ def self.eager_load_rails!
39
+ return if Rails.env.production? && defined?(@rails_eager_loaded)
84
40
 
85
- manifest = if development?
86
- "js.dev.manifest"
87
- else
88
- "js.manifest"
89
- end
90
- @script_manifest = JSON.parse(File.read(root.join(manifest)))
41
+ Rails.application.eager_load! unless Rails.application.config.eager_load
42
+ @rails_eager_loaded = true
91
43
  end
92
44
  end
93
45