filterameter 0.10.0 → 1.0.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 (52) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +106 -56
  3. data/lib/filterameter/configuration.rb +18 -14
  4. data/lib/filterameter/declaration_errors/cannot_be_inline_scope_error.rb +1 -1
  5. data/lib/filterameter/declaration_errors/filter_scope_argument_error.rb +1 -1
  6. data/lib/filterameter/declaration_errors/no_such_attribute_error.rb +1 -1
  7. data/lib/filterameter/declaration_errors/not_a_scope_error.rb +1 -1
  8. data/lib/filterameter/declaration_errors/sort_scope_requires_one_argument_error.rb +1 -1
  9. data/lib/filterameter/declaration_errors/unexpected_error.rb +1 -1
  10. data/lib/filterameter/declaration_errors.rb +1 -1
  11. data/lib/filterameter/declarations_validator.rb +5 -4
  12. data/lib/filterameter/declarative_filters.rb +133 -3
  13. data/lib/filterameter/errors.rb +1 -1
  14. data/lib/filterameter/exceptions/cannot_determine_model_error.rb +1 -1
  15. data/lib/filterameter/exceptions/collection_association_sort_error.rb +3 -3
  16. data/lib/filterameter/exceptions/invalid_association_declaration_error.rb +1 -1
  17. data/lib/filterameter/exceptions/undeclared_parameter_error.rb +1 -1
  18. data/lib/filterameter/exceptions/validation_error.rb +1 -1
  19. data/lib/filterameter/filter_coordinator.rb +1 -1
  20. data/lib/filterameter/filter_declaration.rb +1 -1
  21. data/lib/filterameter/filter_factory.rb +1 -1
  22. data/lib/filterameter/filters/arel_filter.rb +3 -2
  23. data/lib/filterameter/filters/attribute_filter.rb +1 -1
  24. data/lib/filterameter/filters/attribute_validator.rb +1 -1
  25. data/lib/filterameter/filters/conditional_scope_filter.rb +1 -1
  26. data/lib/filterameter/filters/matches_filter.rb +1 -1
  27. data/lib/filterameter/filters/maximum_filter.rb +1 -1
  28. data/lib/filterameter/filters/minimum_filter.rb +1 -1
  29. data/lib/filterameter/filters/nested_collection_filter.rb +1 -1
  30. data/lib/filterameter/filters/nested_filter.rb +1 -1
  31. data/lib/filterameter/filters/scope_filter.rb +1 -1
  32. data/lib/filterameter/helpers/declaration_with_model.rb +1 -1
  33. data/lib/filterameter/helpers/joins_values_builder.rb +1 -1
  34. data/lib/filterameter/helpers/requested_sort.rb +1 -1
  35. data/lib/filterameter/log_subscriber.rb +1 -1
  36. data/lib/filterameter/options/partial_options.rb +9 -8
  37. data/lib/filterameter/parameters_base.rb +1 -1
  38. data/lib/filterameter/query_builder.rb +10 -8
  39. data/lib/filterameter/registries/filter_registry.rb +1 -1
  40. data/lib/filterameter/registries/registry.rb +1 -1
  41. data/lib/filterameter/registries/sort_registry.rb +1 -1
  42. data/lib/filterameter/registries/sub_registry.rb +1 -2
  43. data/lib/filterameter/sort_declaration.rb +1 -1
  44. data/lib/filterameter/sort_factory.rb +1 -1
  45. data/lib/filterameter/sorts/attribute_sort.rb +1 -1
  46. data/lib/filterameter/sorts/scope_sort.rb +1 -1
  47. data/lib/filterameter/validators/inclusion_validator.rb +3 -3
  48. data/lib/filterameter/version.rb +1 -1
  49. data/lib/filterameter.rb +1 -1
  50. metadata +7 -12
  51. data/lib/filterameter/filterable.rb +0 -55
  52. data/lib/filterameter/sortable.rb +0 -40
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 929699bdcf63b640acc335ab2b6690e4d57822541e354f93aa9faf71ef0faf3c
4
- data.tar.gz: 205958bfb5a55418e740382146934b8aa7f3ef122686f903621b6de1f14a2788
3
+ metadata.gz: a250b39d9c80245166bcfe14a7133cd5ec83f9746989d7e461fa0c121358f57a
4
+ data.tar.gz: 5d7c64cb79fd73ee94e4e94316b0b5964fac440b14fb0bc58c0c1d805d0f9d60
5
5
  SHA512:
6
- metadata.gz: dd04092048f8c2d00d4df6f77136e4714e21385b08ca21f0fe1608ba0ff7b12d97d1ca27fc26024f7d52180cb1013546c520131cfe35e90fa688d5c36a185071
7
- data.tar.gz: 1a065d5dcd20ac179e80b1d507ed52de4be84704651cbba2b7fa68c65c0793db06577eec83c68949e880173bfa8ffae930eca2d0ceec68952d26901be7649c93
6
+ metadata.gz: d170eb98f80c24173fa92d0251aa68af7d5f3d002d4c3ff473e086aba8d16f07fd05422af6dced018331e883431c93292b4d0b7e5bf3476587b40df8dbeab8d3
7
+ data.tar.gz: e2342236255b79da8ee96aa7eec0e33c6e9f0131473eaadffd31f833c7212d01d2123eeb6a70cdff8d619ff8678f4d6aafc8d9b52151de9dc6583983f3accd92
data/README.md CHANGED
@@ -4,13 +4,78 @@
4
4
  [![Maintainability](https://api.codeclimate.com/v1/badges/d9d87f9ce8020eb6e656/maintainability)](https://codeclimate.com/github/RockSolt/filterameter/maintainability)
5
5
 
6
6
  # Filterameter
7
- Declarative filter parameters provide clean and clear filters for Rails controllers.
7
+ Filterameter provides declarative filters for Rails controllers to reduce boilerplate code and increase readability. How many times have you seen (or written) this controller action?
8
8
 
9
- ## Usage
10
- Declare filters in controllers to increase readability and reduce boilerplate code. Filters can be declared for attributes or scopes, either directly on the model or on an associated model. Validations can also be assigned.
9
+ ```ruby
10
+ def index
11
+ @films = Films.all
12
+ @films = @films.where(name: params[:name]) if params[:name]
13
+ @films = @films.joins(:film_locations).merge(FilmLocations.where(location_id: params[:location_id])) if params[:location_id]
14
+ @films = @films.directed_by(params[:director_id]) if params[:director_id]
15
+ @films = @films.written_by(params[:writer_id]) if params[:writer_id]
16
+ @films = @films.acted_by(params[:actor_id]) if params[:actor_id]
17
+ end
18
+ ```
11
19
 
12
- Include module `Filterameter::DeclarativeFilters` in the controller to provide the filter DSL. It can be included in the `ApplicationController` to make the functionality available to all controllers or it can be mixed in on a case-by-case basis.
20
+ It's redundant code and a bit of a pain to write and maintain. Not to mention what RuboCop is going to say about it. Wouldn't it be nice if you could just declare the filters that the controller accepts?
13
21
 
22
+ ```ruby
23
+ filter :name, partial: true
24
+ filter :location_id, association: :film_locations
25
+ filter :director_id, name: :directed_by
26
+ filter :writer_id, name: :written_by
27
+ filter :actor_id, name: :acted_by
28
+
29
+ def index
30
+ @films = build_query_from_filters
31
+ end
32
+ ```
33
+
34
+ Simplify and speed development of Rails controllers by making filter parameters declarative with Filterameter.
35
+
36
+ ## Table of Contents
37
+ - [Getting Started](#getting-started)
38
+ - [Usage](#usage)
39
+ - [Filtering Options](#filtering-options)
40
+ - [Name](#name)
41
+ - [Association](#association)
42
+ - [Validates](#validates)
43
+ - [Partial](#partial)
44
+ - [Range](#range)
45
+ - [Sortable](#sortable)
46
+ - [Scope Filters](#scope-filters)
47
+ - [Sorting](#sorting)
48
+ - [Building the Query](#building-the-query)
49
+ - [Specifying the Model](#specifying-the-model)
50
+ - [Configuration](#configuration)
51
+ - [Testing Declarations](#testing-declarations)
52
+ - [Forms and Query Parameters](#forms-and-query-parameters)
53
+ - [Contribute](#contribute)
54
+ - [License](#license)
55
+
56
+ ## Getting Started
57
+
58
+ This gem requires Rails 6.1+, and works with ActiveRecord.
59
+
60
+ ### Installation
61
+ Add this line to your application's Gemfile:
62
+
63
+ ```ruby
64
+ gem 'filterameter'
65
+ ```
66
+
67
+ And then execute:
68
+ ```bash
69
+ $ bundle install
70
+ ```
71
+
72
+ Or install it yourself as:
73
+ ```bash
74
+ $ gem install filterameter
75
+ ```
76
+
77
+ ## Usage
78
+ Include module `Filterameter::DeclarativeFilters` in the controller to provide the filter DSL. It can be included in the `ApplicationController` to make the functionality available to all controllers or it can be mixed in on a case-by-case basis.
14
79
 
15
80
  ```ruby
16
81
  filter :color
@@ -190,18 +255,6 @@ default_sort updated_at: :desc, :description
190
255
 
191
256
  In order to provide consistent results, a sort is always applied. If no default is specified, it will use primary key descending.
192
257
 
193
- ### Specifying the Model
194
-
195
- Rails conventions are used to determine the controller's model. For example, the PhotosController builds a query against the Photo model. If a controller is namespaced, the model will first be looked up without the namespace, then with the namespace.
196
-
197
- **If the conventions do not provide the correct model**, the model can be named explicitly with the following:
198
-
199
- ```ruby
200
- filter_model 'Picture'
201
- ```
202
-
203
- _Important:_ If the `filter_model` declaration is used, it must be before any filter or sort declarations.
204
-
205
258
  ### Building the Query
206
259
 
207
260
  There are two ways to apply the filters and build the query, depending on how much control and/or visibility is desired:
@@ -273,34 +326,29 @@ This method optionally takes a starting query. If there was a controller for Act
273
326
  end
274
327
  ```
275
328
 
276
- Note that the starting query provides the model, so the model is not looked up and any `model_name` declaration is ignored.
277
-
278
- ### Query Parameters
279
-
280
- The query parameters are pulled from the controller parameters, nested under the key `filter`. For example a request for large, blue widgets might have the following url:
281
-
282
- `/widgets?filter[size]=large&filter[color]=blue`
329
+ The starting query is also a good place to provide any includes to enable eager loading:
283
330
 
284
- #### Sort Parameters
285
-
286
- The sort is also nested underneath the key `filter`.
287
-
288
- `/widgets?filter[sort]=size`
331
+ ```ruby
332
+ def index
333
+ @widgets = build_query_from_filters(Widgets.includes(:manufacturer))
334
+ end
335
+ ```
289
336
 
290
- Use an array to pass multiple sorts. The order of the parameters is the order the sorts will be applied. For example, the following sorts first by size then by color:
337
+ Note that the starting query provides the model, so the model is not looked up and the `model_name` declaration in not needed.
291
338
 
292
- `/widgets?filter[sort]=size&filter[sort]=color`
339
+ ### Specifying the Model
293
340
 
294
- Sorts are ascending by default, but can use a prefix can be added to control the sort:
341
+ Rails conventions are used to determine the controller's model. For example, the PhotosController builds a query against the Photo model. If a controller is namespaced, the model will first be looked up without the namespace, then with the namespace.
295
342
 
296
- - `+` ascending (the default)
297
- - `-` descending
343
+ **If the conventions do not provide the correct model**, the model can be named explicitly with the following:
298
344
 
299
- For example, the following sorts by size descending:
345
+ ```ruby
346
+ filter_model 'Picture'
347
+ ```
300
348
 
301
- `/widgets?filter[sort]=-size`
349
+ _Important:_ If the `filter_model` declaration is used, it must be before any filter or sort declarations.
302
350
 
303
- ### Configuration
351
+ ## Configuration
304
352
 
305
353
  There are three configuration options:
306
354
 
@@ -340,7 +388,7 @@ If the filter parameters are NOT nested, set this to false. Doing so will restri
340
388
  those that have been declared, meaning undeclared parameters are ignored (and the action_on_undeclared_parameters
341
389
  configuration option does not come into play).
342
390
 
343
- ### Testing Declarations
391
+ ## Testing Declarations
344
392
 
345
393
  The declarations can be tested for each controller, catching typos, incorrectly defined scopes, or any other issues. Method `declarations_validator` is added to each controller, and a single controller test can be added to validate all the declarations for that controller.
346
394
 
@@ -357,26 +405,9 @@ validator = WidgetsController.declarations_validator
357
405
  assert_predicate validator, :valid?, -> { validator.errors }
358
406
  ```
359
407
 
360
- ## Installation
361
- Add this line to your application's Gemfile:
362
-
363
- ```ruby
364
- gem 'filterameter'
365
- ```
366
-
367
- And then execute:
368
- ```bash
369
- $ bundle
370
- ```
371
-
372
- Or install it yourself as:
373
- ```bash
374
- $ gem install filterameter
375
- ```
376
-
377
408
  ## Forms and Query Parameters
378
409
 
379
- The controller mixin will look for filter parameters nested under the `filter` key. For example, here's what the query parameters might look like for size and color:
410
+ The filter parameters are pulled from the controller parameters, nested under the key `filter` (by default; see [Configuration](#configuration) to change the filter key). For example a request for large, blue widgets might have the following query parameters on the url:
380
411
 
381
412
  ```
382
413
  ?filter[size]=large&filter[color]=blue
@@ -394,7 +425,26 @@ On [a generic search form](https://guides.rubyonrails.org/form_helpers.html#a-ge
394
425
  <% end %>
395
426
  ```
396
427
 
397
- ## Contributions
428
+ #### Sort Parameters
429
+
430
+ The sort is also nested underneath the filter key:
431
+
432
+ `/widgets?filter[sort]=size`
433
+
434
+ Use an array to pass multiple sorts. The order of the parameters is the order the sorts will be applied. For example, the following sorts first by size then by color:
435
+
436
+ `/widgets?filter[sort]=size&filter[sort]=color`
437
+
438
+ Sorts are ascending by default, but can use a prefix can be added to control the sort:
439
+
440
+ - `+` ascending (the default)
441
+ - `-` descending
442
+
443
+ For example, the following sorts by size descending:
444
+
445
+ `/widgets?filter[sort]=-size`
446
+
447
+ ## Contribute
398
448
 
399
449
  Feedback, feature requests, and proposed changes are welcomed. Please use the [issue tracker](https://github.com/RockSolt/filterameter/issues)
400
450
  for feedback and feature requests. To propose a change directly, please fork the repo and open a pull request. Keep an eye on the actions to make
@@ -1,29 +1,33 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Configuration
4
+ # # Configuration
5
5
  #
6
6
  # Class Configuration stores the following settings:
7
- # - action_on_undeclared_parameters
8
- # - action_on_validation_failure
9
- # - filter_key
7
+ # * action_on_undeclared_parameters
8
+ # * action_on_validation_failure
9
+ # * filter_key
10
10
  #
11
- # == Action on Undeclared Parameters
11
+ # ## Action on Undeclared Parameters
12
12
  #
13
- # Occurs when the filter parameter contains any keys that are not defined. Valid actions are :log, :raise, and
14
- # false (do not take action). By default, development will log, test will raise, and production will do nothing.
13
+ # Occurs when the filter parameter contains any keys that are not defined. Valid
14
+ # actions are :log, :raise, and false (do not take action). By default,
15
+ # development will log, test will raise, and production will do nothing.
15
16
  #
16
- # == Action on Validation Failure
17
+ # ## Action on Validation Failure
17
18
  #
18
- # Occurs when a filter parameter fails a validation. Valid actions are :log, :raise, and false (do not take action).
19
- # By default, development will log, test will raise, and production will do nothing.
19
+ # Occurs when a filter parameter fails a validation. Valid actions are :log,
20
+ # :raise, and false (do not take action). By default, development will log, test
21
+ # will raise, and production will do nothing.
20
22
  #
21
- # == Filter Key
23
+ # ## Filter Key
22
24
  #
23
- # By default, the filter parameters are nested under the key :filter. Use this setting to override the key.
25
+ # By default, the filter parameters are nested under the key :filter. Use this
26
+ # setting to override the key.
24
27
  #
25
- # If the filter parameters are NOT nested, set this to false. Doing so will restrict the filter parameters to only
26
- # those that have been declared, meaning undeclared parameters are ignored (and the action_on_undeclared_parameters
28
+ # If the filter parameters are NOT nested, set this to false. Doing so will
29
+ # restrict the filter parameters to only those that have been declared, meaning
30
+ # undeclared parameters are ignored (and the action_on_undeclared_parameters
27
31
  # configuration option does not come into play).
28
32
  class Configuration
29
33
  attr_accessor :action_on_undeclared_parameters, :action_on_validation_failure, :filter_key
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module DeclarationErrors
5
- # = CannotBeInlineScopeError
5
+ # # Cannot Be Inline Scope Error
6
6
  #
7
7
  # Error CannotBeInlineScopeError occurs when an inline scope has been used to define a filter that takes a
8
8
  # parameter. This is not valid for use as a Filterameter filter because an inline scope always has an arity of -1
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module DeclarationErrors
5
- # = Filter Scope Argument Error
5
+ # # Filter Scope Argument Error
6
6
  #
7
7
  # Error FilterScopeArgumentError occurs when a scope used as a filter but does not have either zero or one
8
8
  # arument. A conditional scope filter should take zero arguments; other scope filters should take one argument.
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module DeclarationErrors
5
- # = No Such Attribute Error
5
+ # # No Such Attribute Error
6
6
  #
7
7
  # Error NoSuchAttributeError occurs when a filter or sort references an attribute that does not exist on the model.
8
8
  # The most likely case of this is a typo. Note that if the typo was supposed to reference a scope, this error is
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module DeclarationErrors
5
- # = Not A Scope Error
5
+ # # Not A Scope Error
6
6
  #
7
7
  # Error NotAScopeError flags a class method that has been used as a filter but is not a scope. This could occur if
8
8
  # there is a class method of the same name an attribute, in which case the class method is going to block the
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module DeclarationErrors
5
- # = Sort Scope Requires One Argument Error
5
+ # # Sort Scope Requires One Argument Error
6
6
  #
7
7
  # Error SortScopeRequiresOneArgumentError occurs when a sort has been declared for a scope that does not take
8
8
  # exactly one argument. Sort scopes must take a single argument and will receive either :asc or :desc to indicate
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module DeclarationErrors
5
- # = Unexpected Error
5
+ # # Unexpected Error
6
6
  #
7
7
  # Error UnexpectedError occurs when the filter or scope factory raises an exception that the validator did not
8
8
  # expect.
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module DeclarationErrors
5
- # = Declaration Error
5
+ # # Declaration Error
6
6
  #
7
7
  # Error DeclarationError provides a base class for errors from filter or sort declarations.
8
8
  class DeclarationError < StandardError
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Declarations Validator
4
+ # # Declarations Validator
5
5
  #
6
6
  # Class DeclarationsValidtor fetches each filter and sort from the registry to validate the declaration. This class
7
7
  # can be accessed from the controller as `declarations_validator` (via the FilterCoordinator) and be used in tests.
@@ -10,11 +10,12 @@ module Filterameter
10
10
  #
11
11
  # A test in RSpec might look like this:
12
12
  #
13
- # expect(WidgetsController.declarations_validator).to be_valid
13
+ # expect(WidgetsController.declarations_validator).to be_valid
14
14
  #
15
15
  # In Minitest it might look like this:
16
16
  #
17
- # assert WidgetsController.declarations_validator.valid?, WidgetsController.declarations_validator.errors
17
+ # validator = WidgetsController.declarations_validator
18
+ # assert_predicate validator, :valid?, -> { validator.errors }
18
19
  class DeclarationsValidator
19
20
  include Filterameter::Errors
20
21
 
@@ -69,7 +70,7 @@ module Filterameter
69
70
  "\nInvalid #{type} for '#{name}':\n #{errors.join("\n ")}"
70
71
  end
71
72
 
72
- # = Factory Errors
73
+ # # Factory Errors
73
74
  #
74
75
  # Class FactoryErrors is swapped in if the fetch from a factory fails. It is always invalid and provides the reason.
75
76
  class FactoryErrors
@@ -1,22 +1,134 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Declarative Controller Filters
4
+ # # Declarative Controller Filters
5
5
  #
6
6
  # Mixin DeclarativeFilters should be included in controllers to enable the filter DSL.
7
7
  module DeclarativeFilters
8
8
  extend ActiveSupport::Concern
9
- include Filterameter::Filterable
10
- include Filterameter::Sortable
11
9
 
12
10
  class_methods do
13
11
  delegate :declarations_validator, to: :filter_coordinator
14
12
 
13
+ # Declares a filter that can be read from the parameters and applied to the
14
+ # ActiveRecord query. The `name` identifies the name of the parameter and is the
15
+ # default value to determine the criteria to be applied. The name can be either
16
+ # an attribute or a scope.
17
+ #
18
+ # ## Options
19
+ #
20
+ # The declaration can also include an `options` hash to specialize the behavior of the filter.
21
+ #
22
+ # * **name**: Specify the attribute or scope name if the parameter name is not the same. The default value is the
23
+ # parameter name, so if the two match this can be left out.
24
+ #
25
+ # * **association**: Specify the name of the association if the attribute or scope is nested. Use an array if the
26
+ # asociation is more than one level.
27
+ #
28
+ # * **validates**: Specify a validation if the parameter value should be validated. This uses ActiveModel
29
+ # validations; please review those for types of validations and usage.
30
+ #
31
+ # * **partial**: Specify the partial option if the filter should do a partial search (SQL's `LIKE`). The partial
32
+ # option accepts a hash to specify the search behavior.
33
+ #
34
+ # Here are the available options:
35
+ # * match: `:anywhere` (default), `:from_start`, `:dynamic`
36
+ # * case_sensitive: `true`, `false` (default)
37
+ #
38
+ # There are two shortcuts: the partial option can be declared with `true`,
39
+ # which just uses the defaults; or the partial option can be declared with
40
+ # the match option directly, such as `partial: :from_start`.
41
+ #
42
+ # * **range**: Specify a range option if the filter also allows ranges to be searched. The range option accepts
43
+ # the following options:
44
+ # * true: enables two additional parameters with attribute name plus
45
+ # suffixes `_min` and `_max`
46
+ # * min_only: enables additional parameter with attribute name plus
47
+ # suffix `_min`
48
+ # * max_only: enables additional parameter with attribute name plus
49
+ # suffix `_max`
50
+ #
51
+ # * **sortable**: By default most filters are sortable. To prevent an attribute filter from being sortable, set
52
+ # the option to `false`.
53
+ #
54
+ # Options examples:
55
+ #
56
+ # filter :status, name: :current_status
57
+ # filter :manager_id, association: :department
58
+ # filter :business_unit_name, name: :name, association: [:department, :business_unit]
59
+ # filter :size, validates: { inclusion: { in: %w[Small Medium Large], allow_multiple_values: true } }
60
+ # filter :description, partial: true
61
+ # filter :department_name, partial: :from_start
62
+ # filter :reason, partial: { match: :dynamic, case_sensitive: true }
63
+ # filter :price, range: true
64
+ def filter(name, options = {})
65
+ filter_coordinator.add_filter(name, options)
66
+ end
67
+
68
+ # Declares a list of filters without options. Filters that require options must be declared with `filter`.
69
+ def filters(*names)
70
+ names.each { |name| filter(name) }
71
+ end
72
+
73
+ # Declares a sort that can be read from the parameters and applied to the
74
+ # ActiveRecord query. The `parameter_name` identifies the name of the parameter
75
+ # and is the default value for the attribute name when none is specified in the
76
+ # options.
77
+ #
78
+ # ## Options
79
+ #
80
+ # The declaration can also include an `options` hash to specialize the behavior of the sort.
81
+ #
82
+ # * **name**: Specify the attribute or scope name if the parameter name is not the same. The default value is the
83
+ # parameter name, so if the two match this can be left out.
84
+ #
85
+ # * **association**: Specify the name of the association if the attribute or scope is nested.
86
+ #
87
+ # Options example:
88
+ #
89
+ # sort :project_created_at, name: :created_at, association: :project
90
+ def sort(parameter_name, options = {})
91
+ filter_coordinator.add_sort(parameter_name, options)
92
+ end
93
+
94
+ # Declares a list of sorts without options. Sorts that require options must be declared with `sort`.
95
+ def sorts(*parameter_names)
96
+ parameter_names.each { |parameter_name| filter(parameter_name) }
97
+ end
98
+
99
+ def default_sort(sort_and_direction_pairs)
100
+ filter_coordinator.default_sort = sort_and_direction_pairs
101
+ end
102
+
103
+ # Rails conventions are used to determine the controller's model. For example, the PhotosController builds a query
104
+ # against the Photo model. If a controller is namespaced, the model will first be looked up without the namespace,
105
+ # then with the namespace.
106
+ #
107
+ # **If the conventions do not provide the correct model**, the model can be named explicitly with the following:
108
+ #
109
+ # ```ruby
110
+ # filter_model 'Picture'
111
+ # ```
112
+ #
113
+ # An optional second parameter can be used to specify the variable name. Both the model and the variable name can
114
+ # be specified with this short-cut. For example, to use the Picture model and store the results as `@data`, use
115
+ # the following:
116
+ #
117
+ # filter_model 'Picture', :data
118
+ #
119
+ # **Important:** If the `filter_model` declaration is used, it must be before any filter or sort declarations.
15
120
  def filter_model(model_class, query_var_name = nil)
16
121
  filter_coordinator.model_class = model_class
17
122
  filter_query_var_name(query_var_name) if query_var_name.present?
18
123
  end
19
124
 
125
+ # When using the before action callback `build_filtered_query`, the query is assigned to an instance variable with
126
+ # the name of the model pluralized. For example, the Photo model will use the variable `@photos`.
127
+ #
128
+ # To use a different variable name with the callback, assign it with `filter_query_var_name`. For example, if the
129
+ # query is stored as `@data`, use the following:
130
+ #
131
+ # filter_query_var_name :data
20
132
  def filter_query_var_name(query_variable_name)
21
133
  filter_coordinator.query_variable_name = query_variable_name
22
134
  end
@@ -26,6 +138,24 @@ module Filterameter
26
138
  end
27
139
  end
28
140
 
141
+ # Returns an ActiveRecord query from the filter parameters.
142
+ #
143
+ # def index
144
+ # @widgets = build_query_from_filters
145
+ # end
146
+ #
147
+ # The method optionally takes a starting query. For example, this restricts the results
148
+ # to only active widgets:
149
+ #
150
+ # def index
151
+ # @widgets = build_query_from_filters(Widgets.where(active: true))
152
+ # end
153
+ #
154
+ # The starting query can also be used to provide eager loading:
155
+ #
156
+ # def index
157
+ # @widgets = build_query_from_filters(Widgets.includes(:manufacturer))
158
+ # end
29
159
  def build_query_from_filters(starting_query = nil)
30
160
  self.class.filter_coordinator.build_query(filter_parameters, starting_query)
31
161
  end
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Errors
4
+ # # Errors
5
5
  #
6
6
  # Module Errors provides `valid?` and `errors` to implementing classes. If the `valid?` method is not overridden,
7
7
  # then it returns true.
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Exceptions
5
- # = Cannot Determine Model Error
5
+ # # Cannot Determine Model Error
6
6
  #
7
7
  # Class CannotDetermineModelError is raised when the model class cannot be determined from either the controller
8
8
  # name or controller path. This is a setup issue; the resolution is for the controller to specify the model class
@@ -2,10 +2,10 @@
2
2
 
3
3
  module Filterameter
4
4
  module Exceptions
5
- # = Collection Association Sort Error
5
+ # # Collection Association Sort Error
6
6
  #
7
- # Class CollectionAssociationSortError is raised when a sort is attempted on a collection association. (Sorting is
8
- # only valid on _singular_ associations.)
7
+ # Class CollectionAssociationSortError is raised when a sort is attempted on a
8
+ # collection association. (Sorting is only valid on *singular* associations.)
9
9
  class CollectionAssociationSortError < FilterameterError
10
10
  def initialize(declaration)
11
11
  super("Sorting is not allowed on collection associations: \n\t\t#{declaration}")
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Exceptions
5
- # = Invalid Association Declaration Error
5
+ # # Invalid Association Declaration Error
6
6
  #
7
7
  # Class InvalidAssociationDeclarationError is raised when the declared association(s) are not valid.
8
8
  class InvalidAssociationDeclarationError < FilterameterError
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Exceptions
5
- # = Undeclared Parameter Error
5
+ # # Undeclared Parameter Error
6
6
  #
7
7
  # Class UndeclaredParameterError is raised when a request contains filter parameters that have not been declared.
8
8
  # Configuration setting `action_on_undeclared_parameters` determines whether or not the exception is raised.
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Exceptions
5
- # = Validation Error
5
+ # # Validation Error
6
6
  #
7
7
  # Class ValidationError is raised when a specified parameter fails a validation. Configuration setting
8
8
  # `action_on_validation_failure` determines whether or not the exception is raised.
@@ -8,7 +8,7 @@ require 'action_controller/metal/live'
8
8
  require 'action_controller/metal/strong_parameters'
9
9
 
10
10
  module Filterameter
11
- # = Filter Coordinator
11
+ # # Filter Coordinator
12
12
  #
13
13
  # Class FilterCoordinator stores the configuration declared via class-level method calls such as the list of
14
14
  # filters and the optionally declared model class. Each controller will have one instance of the coordinator
@@ -3,7 +3,7 @@
3
3
  require 'active_support/core_ext/array/wrap'
4
4
 
5
5
  module Filterameter
6
- # = Filter Declaration
6
+ # # Filter Declaration
7
7
  #
8
8
  # Class FilterDeclaration captures the filter declaration within the controller.
9
9
  #
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Filter Factory
4
+ # # Filter Factory
5
5
  #
6
6
  # Class FilterFactory builds a filter from a FilterDeclaration.
7
7
  class FilterFactory
@@ -2,9 +2,10 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Arel Filter
5
+ # # Arel Filter
6
6
  #
7
- # Class ArelFilter is a base class for arel queries. It does not implement <tt>apply</tt>.
7
+ # Class ArelFilter is a base class for arel queries. It does not implement
8
+ # `apply`.
8
9
  class ArelFilter
9
10
  include Filterameter::Errors
10
11
  include Filterameter::Filters::AttributeValidator
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Attribute Filter
5
+ # # Attribute Filter
6
6
  #
7
7
  # Class AttributeFilter leverages ActiveRecord's where query method to add criteria for an attribute.
8
8
  class AttributeFilter
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Attribute Validator
5
+ # # Attribute Validator
6
6
  #
7
7
  # Module AttributeValidator validates that the attribute exists on the model.
8
8
  module AttributeValidator
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Conditional Scope Filter
5
+ # # Conditional Scope Filter
6
6
  #
7
7
  # Class ConditionalScopeFilter applies the scope if the parameter is not false.
8
8
  class ConditionalScopeFilter
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Matches Filter
5
+ # # Matches Filter
6
6
  #
7
7
  # Class MatchesFilter uses arel's `matches` to generate a LIKE query.
8
8
  class MatchesFilter
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Maximum Filter
5
+ # # Maximum Filter
6
6
  #
7
7
  # Class MaximumFilter adds criteria for all values greater than or equal to a maximum.
8
8
  class MaximumFilter < ArelFilter
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Minimum Filter
5
+ # # Minimum Filter
6
6
  #
7
7
  # Class MinimumFilter adds criteria for all values greater than or equal to a minimum.
8
8
  class MinimumFilter < ArelFilter
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Nested Collection Filter
5
+ # # Nested Collection Filter
6
6
  #
7
7
  # Class NestedCollectionFilter joins the nested table(s), merges the filter to the association's model, then makes
8
8
  # the results distinct.
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Nested Attribute Filter
5
+ # # Nested Attribute Filter
6
6
  #
7
7
  # Class NestedFilter joins the nested table(s) then merges the filter to the association's model.
8
8
  class NestedFilter
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Filters
5
- # = Scope Filter
5
+ # # Scope Filter
6
6
  #
7
7
  # Class ScopeFilter applies the named scope passing in the parameter value.
8
8
  class ScopeFilter
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Helpers
5
- # = Declaration with Model
5
+ # # Declaration with Model
6
6
  #
7
7
  # Class DeclarationWithModel inspects the declaration under the context of the model. This enables
8
8
  # predicate methods as well as drilling through associations.
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Helpers
5
- # = Joins Values Builder
5
+ # # Joins Values Builder
6
6
  #
7
7
  # Class JoinsValuesBuilder evaluates an array of names to return either the single entry when there is only
8
8
  # one element in the array or a nested hash when there is more than one element. This is the argument that is
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Helpers
5
- # = Reqested Sort
5
+ # # Reqested Sort
6
6
  #
7
7
  # Class RequestedSort parses the name and direction from a sort segment.
8
8
  class RequestedSort
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Log Subscriber
4
+ # # Log Subscriber
5
5
  #
6
6
  # Class LogSubscriber provides instrumentation for events.
7
7
  class LogSubscriber < ActiveSupport::LogSubscriber
@@ -2,19 +2,20 @@
2
2
 
3
3
  module Filterameter
4
4
  module Options
5
- # = Partial Options
5
+ # # Partial Options
6
6
  #
7
- # Class PartialOptions parses the options passed in as partial, then exposes those. Here are the options along with
8
- # their valid values:
9
- # - match: anywhere (default), from_start, dynamic
10
- # - case_sensitive: true, false (default)
7
+ # Class PartialOptions parses the options passed in as partial, then exposes
8
+ # those. Here are the options along with their valid values:
9
+ # * match: anywhere (default), from_start, dynamic
10
+ # * case_sensitive: true, false (default)
11
11
  #
12
12
  # Options may be specified by passing a hash with the option keys:
13
13
  #
14
- # partial: { match: :from_start, case_sensitive: true }
14
+ # partial: { match: :from_start, case_sensitive: true }
15
15
  #
16
- # There are two shortcuts: the partial option can be declared with `true`, which just uses the defaults; or the
17
- # partial option can be declared with the match option directly, such as partial: :from_start.
16
+ # There are two shortcuts: the partial option can be declared with `true`, which
17
+ # just uses the defaults; or the partial option can be declared with the match
18
+ # option directly, such as partial: :from_start.
18
19
  class PartialOptions
19
20
  VALID_OPTIONS = %i[match case_sensitive].freeze
20
21
  VALID_MATCH_OPTIONS = %w[anywhere from_start dynamic].freeze
@@ -4,7 +4,7 @@ require 'active_model/attribute_assignment'
4
4
  require 'active_model/validations'
5
5
 
6
6
  module Filterameter
7
- # = Parameters
7
+ # # Parameters
8
8
  #
9
9
  # Class Parameters is sub-classed to provide controller-specific validations.
10
10
  class ParametersBase
@@ -1,17 +1,18 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Query Builder
4
+ # # Query Builder
5
5
  #
6
6
  # Class Query Builder turns filter parameters into a query.
7
7
  #
8
8
  # The query builder is instantiated by the filter coordinator. The default query currently is simple `all`. The
9
9
  # default sort comes for the controller declaration of the same name; it is optional and the value may be nil.
10
10
  #
11
- # If the request includes a sort, it is always applied. If not, the following logic kicks in to provide a sort:
12
- # - if the starting query includes a sort, no additional sort is applied
13
- # - if a default sort has been declared, it is applied
14
- # - if neither of those provides a sort, then the fallback is primary key desc
11
+ # If the request includes a sort, it is always applied. If not, the following
12
+ # logic kicks in to provide a sort:
13
+ # * if the starting query includes a sort, no additional sort is applied
14
+ # * if a default sort has been declared, it is applied
15
+ # * if neither of those provides a sort, then the fallback is primary key desc
15
16
  class QueryBuilder
16
17
  def initialize(default_query, default_sort, filter_registry)
17
18
  @default_query = default_query
@@ -87,9 +88,10 @@ module Filterameter
87
88
  end
88
89
  end
89
90
 
90
- # TODO: this handles any runtime exceptions, not just undeclared parameter errors:
91
- # - should the config option be more generalized?
92
- # - or should there be a config option for each type of error?
91
+ # TODO: this handles any runtime exceptions, not just undeclared parameter
92
+ # errors:
93
+ # * should the config option be more generalized?
94
+ # * or should there be a config option for each type of error?
93
95
  def handle_undeclared_parameter(exception)
94
96
  action = Filterameter.configuration.action_on_undeclared_parameters
95
97
  return unless action
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Registries
5
- # Filters
5
+ # # Filter Registry
6
6
  #
7
7
  # Class FilterRegistry is a collection of the filters. It captures the filter declarations when classes are loaded,
8
8
  # then uses the injected FilterFactory to build the filters on demand as they are needed.
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Registries
5
- # = Registry
5
+ # # Registry
6
6
  #
7
7
  # Class Registry records declarations and allows resulting filters and sorts to be fetched from sub-registries.
8
8
  class Registry
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Registries
5
- # Sort Registry
5
+ # # Sort Registry
6
6
  #
7
7
  # Class SortRegistry is a collection of the sorts. It captures the declarations when classes are loaded,
8
8
  # then uses the injected SortFactory to build the sorts on demand as they are needed.
@@ -2,12 +2,11 @@
2
2
 
3
3
  module Filterameter
4
4
  module Registries
5
- # SubRegistry
5
+ # # SubRegistry
6
6
  #
7
7
  # Class SubRegistry provides add and fetch methods as well as the initialization for sub-registries.
8
8
  #
9
9
  # Subclasses must implement build_declaration.
10
- #
11
10
  class SubRegistry
12
11
  def initialize(factory)
13
12
  @factory = factory
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Sort Declaration
4
+ # # Sort Declaration
5
5
  #
6
6
  # Class SortDeclaration captures the sort declaration within the controller. A sort declaration is also generated
7
7
  # from a FilterDeclaration when it is `sortable?`.
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- # = Sort Factory
4
+ # # Sort Factory
5
5
  #
6
6
  # Class SortFactory builds a sort from a model and a declaration.
7
7
  class SortFactory
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Sorts
5
- # = Attribute Sort
5
+ # # Attribute Sort
6
6
  #
7
7
  # Class AttributeSort leverages ActiveRecord's `order` query method to add sorting for an attribute.
8
8
  class AttributeSort
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Filterameter
4
4
  module Sorts
5
- # = Scope Sort
5
+ # # Scope Sort
6
6
  #
7
7
  # Class ScopeSort applies the scope with the a param that is either :asc or :desc. A scope that does not take
8
8
  # exactly one argument is not valid for sorting.
@@ -2,14 +2,14 @@
2
2
 
3
3
  module Filterameter
4
4
  module Validators
5
- # = Inclusion Validator
5
+ # # Inclusion Validator
6
6
  #
7
7
  # Class InclusionValidator extends ActiveModel::Validations::InclusionValidator to enable validations of multiple
8
8
  # values.
9
9
  #
10
- # == Example
10
+ # ## Example
11
11
  #
12
- # validates: { inclusion: { in: %w[Small Medium Large], allow_multiple_values: true } }
12
+ # validates: { inclusion: { in: %w[Small Medium Large], allow_multiple_values: true } }
13
13
  #
14
14
  class InclusionValidator < ActiveModel::Validations::InclusionValidator
15
15
  def validate_each(record, attribute, value)
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Filterameter
4
- VERSION = '0.10.0'
4
+ VERSION = '1.0.1'
5
5
  end
data/lib/filterameter.rb CHANGED
@@ -5,7 +5,7 @@ require 'zeitwerk'
5
5
  loader = Zeitwerk::Loader.for_gem
6
6
  loader.setup # ready!
7
7
 
8
- # = Filterameter
8
+ # # Filterameter
9
9
  module Filterameter
10
10
  class << self
11
11
  attr_writer :configuration
metadata CHANGED
@@ -1,14 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: filterameter
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.10.0
4
+ version: 1.0.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Todd Kummer
8
- autorequire:
9
8
  bindir: bin
10
9
  cert_chain: []
11
- date: 2024-07-29 00:00:00.000000000 Z
10
+ date: 2025-01-09 00:00:00.000000000 Z
12
11
  dependencies:
13
12
  - !ruby/object:Gem::Dependency
14
13
  name: rails
@@ -100,14 +99,14 @@ dependencies:
100
99
  requirements:
101
100
  - - "~>"
102
101
  - !ruby/object:Gem::Version
103
- version: '6.1'
102
+ version: '7.0'
104
103
  type: :development
105
104
  prerelease: false
106
105
  version_requirements: !ruby/object:Gem::Requirement
107
106
  requirements:
108
107
  - - "~>"
109
108
  - !ruby/object:Gem::Version
110
- version: '6.1'
109
+ version: '7.0'
111
110
  - !ruby/object:Gem::Dependency
112
111
  name: rubocop
113
112
  requirement: !ruby/object:Gem::Requirement
@@ -156,14 +155,14 @@ dependencies:
156
155
  requirements:
157
156
  - - "~>"
158
157
  - !ruby/object:Gem::Version
159
- version: 3.0.3
158
+ version: 3.2.0
160
159
  type: :development
161
160
  prerelease: false
162
161
  version_requirements: !ruby/object:Gem::Requirement
163
162
  requirements:
164
163
  - - "~>"
165
164
  - !ruby/object:Gem::Version
166
- version: 3.0.3
165
+ version: 3.2.0
167
166
  - !ruby/object:Gem::Dependency
168
167
  name: rubocop-rspec_rails
169
168
  requirement: !ruby/object:Gem::Requirement
@@ -223,7 +222,6 @@ files:
223
222
  - lib/filterameter/filter_coordinator.rb
224
223
  - lib/filterameter/filter_declaration.rb
225
224
  - lib/filterameter/filter_factory.rb
226
- - lib/filterameter/filterable.rb
227
225
  - lib/filterameter/filters/arel_filter.rb
228
226
  - lib/filterameter/filters/attribute_filter.rb
229
227
  - lib/filterameter/filters/attribute_validator.rb
@@ -247,7 +245,6 @@ files:
247
245
  - lib/filterameter/registries/sub_registry.rb
248
246
  - lib/filterameter/sort_declaration.rb
249
247
  - lib/filterameter/sort_factory.rb
250
- - lib/filterameter/sortable.rb
251
248
  - lib/filterameter/sorts/attribute_sort.rb
252
249
  - lib/filterameter/sorts/scope_sort.rb
253
250
  - lib/filterameter/validators/inclusion_validator.rb
@@ -256,7 +253,6 @@ homepage: https://github.com/RockSolt/filterameter
256
253
  licenses:
257
254
  - MIT
258
255
  metadata: {}
259
- post_install_message:
260
256
  rdoc_options: []
261
257
  require_paths:
262
258
  - lib
@@ -271,8 +267,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
271
267
  - !ruby/object:Gem::Version
272
268
  version: '0'
273
269
  requirements: []
274
- rubygems_version: 3.5.11
275
- signing_key:
270
+ rubygems_version: 3.6.2
276
271
  specification_version: 4
277
272
  summary: Declarative Filter Parameters
278
273
  test_files: []
@@ -1,55 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- module Filterameter
4
- # = Declarative Filters
5
- #
6
- # Mixin Filterable provides class methods <tt>filter</tt> and <tt>filters</tt>.
7
- module Filterable
8
- extend ActiveSupport::Concern
9
-
10
- class_methods do
11
- # Declares a filter that can be read from the parameters and applied to the ActiveRecord query. The <tt>name</tt>
12
- # identifies the name of the parameter and is the default value to determine the criteria to be applied. The name
13
- # can be either an attribute or a scope.
14
- #
15
- # === Options
16
- #
17
- # [:name]
18
- # Specify the attribute or scope name if the parameter name is not the same. The default value
19
- # is the parameter name, so if the two match this can be left out.
20
- #
21
- # [:association]
22
- # Specify the name of the association if the attribute or scope is nested.
23
- #
24
- # [:validates]
25
- # Specify a validation if the parameter value should be validated. This uses ActiveModel validations;
26
- # please review those for types of validations and usage.
27
- #
28
- # [:partial]
29
- # Specify the partial option if the filter should do a partial search (SQL's `LIKE`). The partial
30
- # option accepts a hash to specify the search behavior. Here are the available options:
31
- # - match: anywhere (default), from_start, dynamic
32
- # - case_sensitive: true, false (default)
33
- #
34
- # There are two shortcuts: : the partial option can be declared with `true`, which just uses the
35
- # defaults; or the partial option can be declared with the match option directly,
36
- # such as `partial: :from_start`.
37
- #
38
- # [:range]
39
- # Specify a range option if the filter also allows ranges to be searched. The range option accepts
40
- # the following options:
41
- # - true: enables two additional parameters with attribute name plus suffixes <tt>_min</tt> and <tt>_max</tt>
42
- # - :min_only: enables additional parameter with attribute name plus suffix <tt>_min</tt>
43
- # - :max_only: enables additional parameter with attribute name plus suffix <tt>_max</tt>
44
- def filter(name, options = {})
45
- filter_coordinator.add_filter(name, options)
46
- end
47
-
48
- # Declares a list of filters that can be read from the parameters and applied to the query. The name can be either
49
- # an attribute or a scope. Declare filters individually with <tt>filter</tt> if more options are required.
50
- def filters(*names)
51
- names.each { |name| filter(name) }
52
- end
53
- end
54
- end
55
- end
@@ -1,40 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- module Filterameter
4
- # = Sortable
5
- #
6
- # Mixin Sortable provides class methods <tt>sort</tt> and <tt>sorts</tt>.
7
- module Sortable
8
- extend ActiveSupport::Concern
9
-
10
- class_methods do
11
- # Declares a sort that can be read from the parameters and applied to the ActiveRecord query. The
12
- # <tt>parameter_name</tt> identifies the name of the parameter and is the default value for the attribute
13
- # name when none is specified in the options.
14
- #
15
- # The including class must implement `filter_coordinator`
16
- #
17
- # === Options
18
- #
19
- # [:name]
20
- # Specify the attribute or scope name if the parameter name is not the same. The default value
21
- # is the parameter name, so if the two match this can be left out.
22
- #
23
- # [:association]
24
- # Specify the name of the association if the attribute or scope is nested.
25
- def sort(parameter_name, options = {})
26
- filter_coordinator.add_sort(parameter_name, options)
27
- end
28
-
29
- # Declares a list of filters that can be read from the parameters and applied to the query. The name can be either
30
- # an attribute or a scope. Declare filters individually with <tt>filter</tt> if more options are required.
31
- def sorts(*parameter_names)
32
- parameter_names.each { |parameter_name| filter(parameter_name) }
33
- end
34
-
35
- def default_sort(sort_and_direction_pairs)
36
- filter_coordinator.default_sort = sort_and_direction_pairs
37
- end
38
- end
39
- end
40
- end