base_editing_bootstrap 1.6.0 → 1.8.0
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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +38 -0
- data/README.md +2 -1
- data/app/controllers/base_editing_controller.rb +17 -5
- data/app/helpers/utilities/form_helper.rb +13 -2
- data/app/helpers/utilities/page_helper.rb +7 -3
- data/app/helpers/utilities/template_helper.rb +15 -14
- data/app/policies/base_model_policy.rb +2 -0
- data/app/views/base_editing/_editing_form_help_text.html.erb +7 -1
- data/app/views/base_editing/_form.html.erb +16 -17
- data/app/views/base_editing/_form_base_errors.html.erb +22 -0
- data/app/views/base_editing/_form_body_container.html.erb +3 -0
- data/app/views/base_editing/_form_container.html.erb +3 -0
- data/app/views/base_editing/_form_field.html.erb +6 -6
- data/app/views/base_editing/_form_footer_container.html.erb +3 -0
- data/app/views/base_editing/form_field/_textarea.html.erb +7 -0
- data/config/locales/it.yml +7 -0
- data/lib/base_editing_bootstrap/VERSION +1 -1
- data/lib/base_editing_bootstrap/base_model.rb +11 -1
- data/lib/base_editing_bootstrap/forms/base.rb +10 -9
- data/lib/base_editing_bootstrap/searches/base.rb +4 -3
- data/lib/generators/base_editing_bootstrap/scaffold/scaffold_generator.rb +1 -1
- data/lib/generators/base_editing_bootstrap/scaffold/templates/spec/model.rb.tt +4 -1
- data/spec/support/external_shared/base_editing_controller_helpers.rb +11 -6
- data/spec/support/external_shared/base_model.rb +17 -2
- data/spec/support/external_shared/pundit.rb +1 -0
- metadata +7 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 785488fc3726ebf47d41122c81fd74cfc1d25dcef6b5f3848d80689a5eb00e5b
|
4
|
+
data.tar.gz: e63fa0c1c288e38fde44df84260eefd133d726ae387e8742f549753a83394959
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 403dc430a15bdcb9030900197c2acc309fe2108a2a5c68340eed924a76264d2b57086022fa88be5bb2abc5b6b87acaaf62574c1c1d086a1cb5682afaddef20c0
|
7
|
+
data.tar.gz: f0b42845d080d3fc4d52d5eeca2edf35bca3cb4610bac33655b69c17e135c10b7fb96ee6fe0599cfdca66c645b30b03b50b76aa7953646f089d7bfb14e6832ed
|
data/CHANGELOG.md
CHANGED
@@ -2,6 +2,44 @@
|
|
2
2
|
All notable changes to this project will be documented in this file. See [conventional commits](https://www.conventionalcommits.org/) for commit guidelines.
|
3
3
|
|
4
4
|
- - -
|
5
|
+
## 1.8.0 - 2025-03-20
|
6
|
+
#### Bug Fixes
|
7
|
+
- Correct pass locals datas - (00fd083) - Marino Bonetti
|
8
|
+
- Don't re-authorize if authorization is already done - (dd756b3) - Marino Bonetti
|
9
|
+
- Correct path for generator - (a7d9842) - Marino Bonetti
|
10
|
+
- Search partial with class name and without namespace - (2d08db3) - Marino Bonetti
|
11
|
+
- Belongs to polymorphic (#17) - (7d736d7) - Marino Bonetti
|
12
|
+
#### Documentation
|
13
|
+
- Update documentation - (07dd44d) - Marino Bonetti
|
14
|
+
- Correct documentation - (6337745) - Marino Bonetti
|
15
|
+
#### Features
|
16
|
+
- Aggiunto contenitore per gli errori base - (2f9f309) - Marino Bonetti
|
17
|
+
- Add layouts for customization - (1bb0201) - Marino Bonetti
|
18
|
+
- AutoRender text columns to TextArea - (8533b53) - Marino Bonetti
|
19
|
+
#### Refactoring
|
20
|
+
- Rename variable to be more logic - (8fa9bbb) - Marino Bonetti
|
21
|
+
- Rewrite for debugging - (b7a3756) - Marino Bonetti
|
22
|
+
#### Tests
|
23
|
+
- Typo - (a1ada44) - Marino Bonetti
|
24
|
+
|
25
|
+
- - -
|
26
|
+
|
27
|
+
## 1.7.0 - 2025-02-11
|
28
|
+
#### Bug Fixes
|
29
|
+
- Add permits of ransack scopes - (5fa26c6) - Marino Bonetti
|
30
|
+
- Correct set select classes - (a5ef685) - Marino Bonetti
|
31
|
+
- Boolean_icon with nil value - (211c91f) - Marino Bonetti
|
32
|
+
- Better expection for spec helper - (65b26ce) - Marino Bonetti
|
33
|
+
#### Documentation
|
34
|
+
- Add documentation for help html version - (dbc1c01) - Marino Bonetti
|
35
|
+
#### Features
|
36
|
+
- Add HTML helper version in form field help - (f06c438) - Marino Bonetti
|
37
|
+
- Add Configurable distinct query in controller - (184d772) - Marino Bonetti
|
38
|
+
#### Tests
|
39
|
+
- Correct validation for selects - (e4b8016) - Marino Bonetti
|
40
|
+
|
41
|
+
- - -
|
42
|
+
|
5
43
|
## 1.6.0 - 2025-01-23
|
6
44
|
#### Features
|
7
45
|
- Add new test helper for association relation - (15690c7) - Marino Bonetti
|
data/README.md
CHANGED
@@ -126,7 +126,8 @@ Utilizzo per modello base, in questo esempio prendiamo come modello Post come es
|
|
126
126
|
```
|
127
127
|
- è possibile customizzare
|
128
128
|
- un text help per ogni campo andando ad aggiungere nelle traduzioni la relativa
|
129
|
-
traduzione nella posizione: `it.activerecord.attributes.MODEL.FIELD/help_text`
|
129
|
+
traduzione nella posizione: `it.activerecord.attributes.MODEL.FIELD/help_text` oppure `help_text_html` in caso di
|
130
|
+
contenuto con html
|
130
131
|
- un blocco per l'unità di misura accanto al campo aggiungendo alle traduzioni:
|
131
132
|
`it.activerecord.attributes.MODEL.FIELD/unit`
|
132
133
|
|
@@ -14,13 +14,20 @@ class BaseEditingController < RestrictedAreaController
|
|
14
14
|
# Works like documented in https://activerecord-hackery.github.io/ransack/getting-started/sorting/#sorting-in-the-controller
|
15
15
|
class_attribute :default_sorts, default: ["id"]
|
16
16
|
|
17
|
+
##
|
18
|
+
# Configure default distinct results in the index query.
|
19
|
+
# Works like documented in https://activerecord-hackery.github.io/ransack/going-further/other-notes/#problem-with-distinct-selects
|
20
|
+
class_attribute :default_distinct, default: true
|
21
|
+
|
17
22
|
def index
|
18
|
-
|
23
|
+
#se è già stato autorizzano non rieseguiamo, utile nel caso vogliamo sovrascrivere la logica di autorizzazione in inheritance
|
24
|
+
authorize base_class unless pundit_policy_authorized?
|
19
25
|
|
20
26
|
q = policy_scope(base_scope)
|
21
27
|
@search_instance = search_class.new(q, current_user,
|
22
28
|
params: params.permit(:page, :q => {}), # FIXME trovare modo per essere più "STRONG"
|
23
|
-
sorts: default_sorts
|
29
|
+
sorts: default_sorts,
|
30
|
+
distinct: default_distinct
|
24
31
|
)
|
25
32
|
@search_instance = yield(@search_instance) if block_given?
|
26
33
|
end
|
@@ -28,7 +35,9 @@ class BaseEditingController < RestrictedAreaController
|
|
28
35
|
def new
|
29
36
|
@object = base_class.new
|
30
37
|
@object = yield(@object) if block_given?
|
31
|
-
|
38
|
+
|
39
|
+
#se è già stato autorizzano non rieseguiamo, utile nel caso vogliamo sovrascrivere la logica di autorizzazione in inheritance
|
40
|
+
authorize @object unless pundit_policy_authorized?
|
32
41
|
|
33
42
|
respond_to do |format|
|
34
43
|
format.html
|
@@ -58,7 +67,8 @@ class BaseEditingController < RestrictedAreaController
|
|
58
67
|
def create
|
59
68
|
@object = base_class.new(permitted_attributes)
|
60
69
|
@object = yield(@object) if block_given?
|
61
|
-
|
70
|
+
#se è già stato autorizzano non rieseguiamo, utile nel caso vogliamo sovrascrivere la logica di autorizzazione in inheritance
|
71
|
+
authorize @object unless pundit_policy_authorized?
|
62
72
|
|
63
73
|
respond_to do |format|
|
64
74
|
if @object.save
|
@@ -118,7 +128,9 @@ class BaseEditingController < RestrictedAreaController
|
|
118
128
|
|
119
129
|
def load_object
|
120
130
|
@object = base_class.find(params[:id])
|
121
|
-
|
131
|
+
|
132
|
+
#se è già stato autorizzano non rieseguiamo, utile nel caso vogliamo sovrascrivere la logica di autorizzazione in inheritance
|
133
|
+
authorize @object unless pundit_policy_authorized?
|
122
134
|
logger.debug { "Oggetto #{@object.inspect}" }
|
123
135
|
end
|
124
136
|
|
@@ -24,7 +24,8 @@ module Utilities
|
|
24
24
|
type = :enum
|
25
25
|
generic_field = "enum"
|
26
26
|
elsif form.object.class.respond_to?(:reflect_on_association) &&
|
27
|
-
form.object.class.reflect_on_association(field.to_s).is_a?(ActiveRecord::Reflection::BelongsToReflection)
|
27
|
+
form.object.class.reflect_on_association(field.to_s).is_a?(ActiveRecord::Reflection::BelongsToReflection) &&
|
28
|
+
!form.object.class.reflect_on_association(field.to_s).polymorphic? # non deve essere polymorphic
|
28
29
|
# Abbiamo una relazione belongs_to da gestire
|
29
30
|
reflection = form.object.class.reflect_on_association(field.to_s)
|
30
31
|
type = :belongs_to
|
@@ -60,6 +61,8 @@ module Utilities
|
|
60
61
|
generic_field = "boolean"
|
61
62
|
when :has_one_attachment
|
62
63
|
generic_field = "has_one_attachment"
|
64
|
+
when :text
|
65
|
+
generic_field = "textarea"
|
63
66
|
else
|
64
67
|
generic_field = "base"
|
65
68
|
end
|
@@ -71,7 +74,15 @@ module Utilities
|
|
71
74
|
"form_field",
|
72
75
|
generic_field
|
73
76
|
)
|
74
|
-
Rails.logger.debug
|
77
|
+
Rails.logger.debug do
|
78
|
+
<<~TEXT
|
79
|
+
[BASE EDITING BOOTSTRAP]
|
80
|
+
TYPE: #{type}
|
81
|
+
GENERIC_FIELD: #{generic_field}
|
82
|
+
TEMPLATE: #{template.short_identifier}
|
83
|
+
LOCALS:#{locals}
|
84
|
+
TEXT
|
85
|
+
end
|
75
86
|
template.render(self, locals)
|
76
87
|
end
|
77
88
|
|
@@ -1,4 +1,5 @@
|
|
1
1
|
module Utilities::PageHelper
|
2
|
+
include Utilities::IconHelper
|
2
3
|
# @param [BaseModel] base_class
|
3
4
|
def title_mod_g(base_class)
|
4
5
|
"#{t("edit")} #{base_class.model_name.human}"
|
@@ -24,12 +25,15 @@ module Utilities::PageHelper
|
|
24
25
|
# end
|
25
26
|
# end
|
26
27
|
|
27
|
-
# @param [TrueClass, FalseClass] valore
|
28
|
+
# @param [TrueClass, FalseClass, NilClass] valore
|
28
29
|
def boolean_to_icon(valore)
|
29
|
-
|
30
|
+
case valore
|
31
|
+
when true
|
30
32
|
icon("check-lg", class: "text-success")
|
31
|
-
|
33
|
+
when false
|
32
34
|
icon("x-lg", class: "text-danger")
|
35
|
+
else
|
36
|
+
nil
|
33
37
|
end
|
34
38
|
end
|
35
39
|
|
@@ -18,21 +18,22 @@ module Utilities::TemplateHelper
|
|
18
18
|
partial_path = (obj.respond_to? :to_partial_path) ? obj.to_partial_path : obj._to_partial_path
|
19
19
|
obj_base_path = "#{partial_path}/#{base_path}"
|
20
20
|
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
21
|
+
[
|
22
|
+
# Precedenza modello e campo specifico
|
23
|
+
[field, [obj_base_path]],
|
24
|
+
# cerco tramite nome modello semplice, con namespace della risorsa (cell_field,header_field,form_field) e nome del campo specifico
|
25
|
+
["#{obj.model_name.element}/#{base_path}/#{field}", lookup_context.prefixes],
|
26
|
+
# Ricerca tramite campo generico e prefissi di contesto che contiene anche controller e namespace di controller
|
27
|
+
["#{base_path}/#{generic_field}", lookup_context.prefixes],
|
28
|
+
[generic_field, [obj_base_path]],
|
29
|
+
["base_editing/#{base_path}/#{generic_field}", []],
|
30
|
+
].each do |partial, prefixes|
|
31
|
+
Rails.logger.debug { "[BASE EDITING BOOTSTRAP] Cerco partial:`#{partial}` in #{prefixes.inspect}" }
|
32
|
+
if lookup_context.exists?(partial, prefixes, true)
|
33
|
+
return lookup_context.find(partial, prefixes, true)
|
34
|
+
end
|
35
35
|
end
|
36
|
+
# fallback finale
|
36
37
|
lookup_context.find("base_editing/#{base_path}/base", [], true)
|
37
38
|
end
|
38
39
|
|
@@ -6,7 +6,13 @@
|
|
6
6
|
%>
|
7
7
|
<%# locals: (object:,field:) -%>
|
8
8
|
<%
|
9
|
-
|
9
|
+
nf = "NOT_FOUND_HTML"
|
10
|
+
help_text = object.class.human_attribute_name("#{field}/help_text_html", default: nf)
|
11
|
+
if help_text != nf
|
12
|
+
help_text = help_text.html_safe
|
13
|
+
else
|
14
|
+
help_text = object.class.human_attribute_name("#{field}/help_text", default: "")
|
15
|
+
end
|
10
16
|
unless help_text.blank?
|
11
17
|
%>
|
12
18
|
<div class="form-text"><%= help_text %></div>
|
@@ -1,20 +1,19 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
<%= form_for @object, builder: form_builder do |form| %>
|
1
|
+
<%= render layout: "form_container" do %>
|
2
|
+
<%= form_for @object, builder: form_builder do |form| %>
|
4
3
|
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
<div class="card-footer">
|
15
|
-
<%= render partial: "form_footer", locals: {form: form} %>
|
16
|
-
</div>
|
4
|
+
<%= render layout: "form_body_container" do %>
|
5
|
+
<%= render partial: "form_field_header", locals: {form:} %>
|
6
|
+
<%= render partial: "form_base_errors", locals: {form:} if form.object.errors.key?(:base) %>
|
7
|
+
<%= render layout: "form_fields_container" do %>
|
8
|
+
<%= render collection: form_attributes(form.object),
|
9
|
+
layout: "form_field_container",
|
10
|
+
partial: "form_field", locals: {form:} %>
|
11
|
+
<% end %>
|
12
|
+
<% end %>
|
17
13
|
|
14
|
+
<%= render layout: "form_footer_container" do %>
|
15
|
+
<%= render partial: "form_footer", locals: {form: form} %>
|
18
16
|
<% end %>
|
19
|
-
|
20
|
-
|
17
|
+
|
18
|
+
<% end %>
|
19
|
+
<% end %>
|
@@ -0,0 +1,22 @@
|
|
1
|
+
<%
|
2
|
+
##
|
3
|
+
# Template per il rendering degli errori base
|
4
|
+
# - form -> FormBuilder
|
5
|
+
%>
|
6
|
+
<%# locals: (form:) -%>
|
7
|
+
<% if form.object.errors.key?(:base) %>
|
8
|
+
<div class="row base-errors-container" id="<%= dom_id(@object,:base_errors_container) %>">
|
9
|
+
<div class="col-12">
|
10
|
+
<div class="card border-danger mb-3">
|
11
|
+
<div class="card-header"><%= t ".title", scope: form.object.class.model_name.i18n_key, default: t(".title") %></div>
|
12
|
+
<div class="card-body text-danger">
|
13
|
+
<ul>
|
14
|
+
<% form.object.errors.full_messages_for(:base).each do |m| %>
|
15
|
+
<li><%= m %></li>
|
16
|
+
<% end %>
|
17
|
+
</ul>
|
18
|
+
</div>
|
19
|
+
</div>
|
20
|
+
</div>
|
21
|
+
</div>
|
22
|
+
<% end %>
|
@@ -2,18 +2,18 @@
|
|
2
2
|
##
|
3
3
|
# Template per il rendering campo
|
4
4
|
# - form -> FormBuilder
|
5
|
-
# - form_field ->
|
5
|
+
# - form_field -> String
|
6
6
|
%>
|
7
7
|
<%# locals: (form:, form_field:) -%>
|
8
8
|
<%
|
9
|
-
|
9
|
+
input_group_classes = ["mb-1"]
|
10
10
|
|
11
11
|
content = form_print_field(form, form_field)
|
12
12
|
unless content.match "checkbox"
|
13
|
-
|
13
|
+
input_group_classes << "input-group"
|
14
14
|
end
|
15
|
-
|
16
|
-
|
15
|
+
input_group_classes << (form.object.validated? ? "has-validation" : "")
|
16
|
+
input_group_classes << "form-#{form_field}-input-group"
|
17
17
|
|
18
18
|
# Non renderizziamo il contenuto del label in questa vista se siamo nello switch
|
19
19
|
if content.match "form-switch"
|
@@ -24,7 +24,7 @@
|
|
24
24
|
|
25
25
|
%>
|
26
26
|
<%= label_content %>
|
27
|
-
<%= content_tag :div, class:
|
27
|
+
<%= content_tag :div, class: input_group_classes.join(" ") do %>
|
28
28
|
<%= render partial: "editing_form_measure_unit", locals: {object: form.object, field: form_field} %>
|
29
29
|
<%= content %>
|
30
30
|
<%= error_messages_for(form.object, form_field) %>
|
data/config/locales/it.yml
CHANGED
@@ -1 +1 @@
|
|
1
|
-
1.
|
1
|
+
1.8.0
|
@@ -7,7 +7,9 @@ module BaseEditingBootstrap
|
|
7
7
|
included do
|
8
8
|
include IsValidated
|
9
9
|
include ActionTranslation
|
10
|
-
delegate :ransackable_attributes,
|
10
|
+
delegate :ransackable_attributes,
|
11
|
+
:ransackable_associations,
|
12
|
+
:ransackable_scopes, to: :@class
|
11
13
|
|
12
14
|
|
13
15
|
##
|
@@ -34,6 +36,14 @@ module BaseEditingBootstrap
|
|
34
36
|
Pundit.policy(User.new, self.new).permitted_associations_for_ransack.map(&:to_s)
|
35
37
|
end
|
36
38
|
end
|
39
|
+
|
40
|
+
def ransackable_scopes(auth_object = nil)
|
41
|
+
if auth_object
|
42
|
+
Pundit.policy(auth_object, self.new).permitted_scopes_for_ransack.map(&:to_s)
|
43
|
+
else
|
44
|
+
Pundit.policy(User.new, self.new).permitted_scopes_for_ransack.map(&:to_s)
|
45
|
+
end
|
46
|
+
end
|
37
47
|
end
|
38
48
|
end
|
39
49
|
end
|
@@ -24,8 +24,8 @@ module BaseEditingBootstrap::Forms
|
|
24
24
|
##
|
25
25
|
# Costruisce l'array delle classi che devono essere presenti sul campo della form
|
26
26
|
#
|
27
|
-
def form_style_class_for(method, options = {})
|
28
|
-
classes =
|
27
|
+
def form_style_class_for(method, options = {}, base_classes: ["form-control"])
|
28
|
+
classes = base_classes
|
29
29
|
classes << "is-invalid" if object.errors && object.errors.include?(method)
|
30
30
|
classes << options[:class].split(" ") if options[:class]
|
31
31
|
classes.flatten.compact.uniq.join(" ")
|
@@ -46,8 +46,8 @@ module BaseEditingBootstrap::Forms
|
|
46
46
|
end
|
47
47
|
|
48
48
|
def select(method, choices = nil, options = {}, html_options = {}, &block)
|
49
|
-
html_options
|
50
|
-
super(method, choices, options,
|
49
|
+
html_options.merge!(class: form_style_class_for(method, html_options, base_classes: ["form-control", "form-select"]))
|
50
|
+
super(method, choices, options,html_options , &block)
|
51
51
|
end
|
52
52
|
|
53
53
|
def check_box(method, options = {}, checked_value = "1", unchecked_value = "0")
|
@@ -57,9 +57,10 @@ module BaseEditingBootstrap::Forms
|
|
57
57
|
else
|
58
58
|
label_tag = nil
|
59
59
|
end
|
60
|
-
|
61
|
-
|
62
|
-
|
60
|
+
@template.content_tag(:div, class: form_style_class_for(method, {class: (options.extract!(:class)[:class] || "")}, base_classes: ["form-check"])) do
|
61
|
+
checkbox_class = ["form-check-input"]
|
62
|
+
checkbox_class << "is-invalid" if object.errors && object.errors.include?(method)
|
63
|
+
super(method, options.reverse_merge(class: checkbox_class.join(" ")), checked_value, unchecked_value) + label_tag
|
63
64
|
end
|
64
65
|
end
|
65
66
|
|
@@ -75,7 +76,7 @@ module BaseEditingBootstrap::Forms
|
|
75
76
|
options = {},
|
76
77
|
html_options = {},
|
77
78
|
&block)
|
78
|
-
form_check_classes = (["form-check"] + [(html_options.delete(:form_check_class){""}).split(" ").collect(&:strip)]).compact.join(" ")
|
79
|
+
form_check_classes = (["form-check"] + [(html_options.delete(:form_check_class) { "" }).split(" ").collect(&:strip)]).compact.join(" ")
|
79
80
|
super do |builder|
|
80
81
|
@template.content_tag(:div, class: form_check_classes) do
|
81
82
|
builder.check_box(class: "form-check-input") + builder.label(class: "form-check-label")
|
@@ -96,7 +97,7 @@ module BaseEditingBootstrap::Forms
|
|
96
97
|
# per il normale submit consiglio la lettura della guida standard di rails
|
97
98
|
# ATTENZIONE: nelle classi del bottone undo, abbiamo aggiunto .btn-undo-button
|
98
99
|
# che ascoltiamo dalle modal e utilizziamo per chiudere la modal, al posto
|
99
|
-
# seguire realmente il link con il browser.
|
100
|
+
# di seguire realmente il link con il browser.
|
100
101
|
def submit(value = nil, options = {})
|
101
102
|
@template.content_tag(:div, class: "btn-group mr-1") do
|
102
103
|
super(value, options.reverse_merge(class: "btn btn-primary")) +
|
@@ -7,17 +7,18 @@ module BaseEditingBootstrap::Searches
|
|
7
7
|
include ActiveModel::Naming
|
8
8
|
include ActiveModel::Conversion
|
9
9
|
|
10
|
-
attr_reader :model_klass, :user, :params, :scope, :sorts
|
10
|
+
attr_reader :model_klass, :user, :params, :scope, :sorts, :distinct
|
11
11
|
|
12
12
|
# @param [User] user
|
13
13
|
# @param [ActiveRecord::Associations::CollectionProxy] scope
|
14
14
|
# @param [Array<String (frozen)>] sort
|
15
|
-
def initialize(scope, user, params: {page: nil}, sorts: ["id"])
|
15
|
+
def initialize(scope, user, params: {page: nil}, sorts: ["id"], distinct: true)
|
16
16
|
@model_klass = scope.klass
|
17
17
|
@user = user
|
18
18
|
@scope = scope
|
19
19
|
@params = params
|
20
20
|
@sorts = sorts
|
21
|
+
@distinct = distinct
|
21
22
|
end
|
22
23
|
|
23
24
|
##
|
@@ -26,7 +27,7 @@ module BaseEditingBootstrap::Searches
|
|
26
27
|
def results
|
27
28
|
ransack_query
|
28
29
|
.tap { |r| r.sorts = @sorts if r.sorts.empty? }
|
29
|
-
.result(distinct:
|
30
|
+
.result(distinct: @distinct)
|
30
31
|
.tap { |q| Rails.logger.debug { "[Ransack] params:#{params} - sql: #{q.to_sql}" } }
|
31
32
|
.page(params[:page])
|
32
33
|
end
|
@@ -11,7 +11,7 @@ module BaseEditingBootstrap
|
|
11
11
|
invoke :model
|
12
12
|
|
13
13
|
def add_base_model
|
14
|
-
inject_into_class "app/models/#{
|
14
|
+
inject_into_class "app/models/#{file_path}.rb", class_name do
|
15
15
|
" include BaseEditingBootstrap::BaseModel\n"
|
16
16
|
end
|
17
17
|
|
@@ -5,11 +5,14 @@ RSpec.describe <%= class_name %>, type: :model do
|
|
5
5
|
|
6
6
|
# it_behaves_like "a base model",
|
7
7
|
# ransack_permitted_attributes: %w[<%= attributes_names.join(" ") %>],
|
8
|
-
# ransack_permitted_associations: []
|
8
|
+
# ransack_permitted_associations: [],
|
9
|
+
# option_label_method: :to_s,
|
10
|
+
# ransack_permitted_scopes: [] do
|
9
11
|
# let(:auth_object) { :auth_object } <- default
|
10
12
|
# let(:auth_object) { create(:user, :as_admin) } <- in caso di necessità di override
|
11
13
|
# let(:new_user_ransack_permitted_attributes) { ransack_permitted_attributes }
|
12
14
|
# let(:new_user_ransack_permitted_associations) { ransack_permitted_associations }
|
15
|
+
# let(:new_user_ransack_permitted_scopes) { ransack_permitted_scopes }
|
13
16
|
# end
|
14
17
|
|
15
18
|
end
|
@@ -49,7 +49,9 @@ RSpec.shared_examples "base editing controller" do |factory: nil, only: [], exce
|
|
49
49
|
|
50
50
|
##
|
51
51
|
# Possibili override per la costruzione delle path
|
52
|
-
|
52
|
+
|
53
|
+
let(:default_sorts) { BaseEditingController.default_sorts } # configurazione nel controller per ordine di default del modello
|
54
|
+
let(:default_distinct) { BaseEditingController.default_distinct } # configurazione nel controller per distinct nella ricerca di ransack
|
53
55
|
let(:url_for_new) { url_for([model.new, action: :new]) }
|
54
56
|
let(:url_for_index) { url_for(model) }
|
55
57
|
let(:url_for_create) { url_for(model.new) }
|
@@ -89,10 +91,13 @@ RSpec.shared_examples "base editing controller" do |factory: nil, only: [], exce
|
|
89
91
|
params = {q: {"foo_eq": "foo"}}
|
90
92
|
get url_for_index, params: params
|
91
93
|
expect(response).to have_http_status(200)
|
92
|
-
expect(assigns[:search_instance]).to be_an_instance_of(BaseEditingBootstrap::Searches::Base)
|
93
|
-
|
94
|
-
|
95
|
-
|
94
|
+
expect(assigns[:search_instance]).to be_an_instance_of(BaseEditingBootstrap::Searches::Base)
|
95
|
+
.and(have_attributes(
|
96
|
+
user: user,
|
97
|
+
params: ActionController::Parameters.new(params).permit!,
|
98
|
+
sorts: default_sorts,
|
99
|
+
distinct: default_distinct,
|
100
|
+
))
|
96
101
|
end
|
97
102
|
end
|
98
103
|
end
|
@@ -194,7 +199,7 @@ default_unathorized_failure = -> { raise "TODO - passare proc con richiesta che
|
|
194
199
|
RSpec.shared_examples "fail with unauthorized" do |request: default_unathorized_failure|
|
195
200
|
it "is expected to redirect to root" do
|
196
201
|
|
197
|
-
if Gem::Version.create(
|
202
|
+
if Gem::Version.create(Pundit::VERSION) < Gem::Version.create('2.3.2')
|
198
203
|
allow(Pundit).to receive(:authorize).with(user, any_args).and_raise(Pundit::NotAuthorizedError)
|
199
204
|
else
|
200
205
|
allow_any_instance_of(Pundit::Context).to receive(:authorize).and_raise(Pundit::NotAuthorizedError)
|
@@ -1,4 +1,7 @@
|
|
1
|
-
RSpec.shared_examples "a base model" do |ransack_permitted_attributes: [],
|
1
|
+
RSpec.shared_examples "a base model" do |ransack_permitted_attributes: [],
|
2
|
+
ransack_permitted_associations: [],
|
3
|
+
option_label_method: :to_s,
|
4
|
+
ransack_permitted_scopes: []|
|
2
5
|
|
3
6
|
it_behaves_like "a validated? object"
|
4
7
|
|
@@ -10,15 +13,24 @@ RSpec.shared_examples "a base model" do |ransack_permitted_attributes: [], ransa
|
|
10
13
|
instance = described_class.new
|
11
14
|
expect(instance).to respond_to(:option_label)
|
12
15
|
|
13
|
-
expect(instance).to receive(option_label_method).and_call_original
|
16
|
+
expect(instance).to receive(option_label_method).and_call_original, "Expected `#{instance.class}#option_label` chiami il metodo `##{option_label_method}` per la traduzione del label nelle options"
|
14
17
|
instance.option_label
|
15
18
|
end
|
16
19
|
|
20
|
+
if ransack_permitted_scopes.any?
|
21
|
+
it "have scopes" do
|
22
|
+
ransack_permitted_scopes.each do |scope|
|
23
|
+
expect(described_class).to respond_to(scope)
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
17
28
|
##
|
18
29
|
# Oggetto solitamente di classe User che identifichi l'utente a cui eseguire il check dei permessi
|
19
30
|
let(:auth_object) { :auth_object }
|
20
31
|
let(:new_user_ransack_permitted_attributes) { ransack_permitted_attributes }
|
21
32
|
let(:new_user_ransack_permitted_associations) { ransack_permitted_associations }
|
33
|
+
let(:new_user_ransack_permitted_scopes) { ransack_permitted_scopes }
|
22
34
|
|
23
35
|
describe "with ransackables" do
|
24
36
|
where(:base_model_method, :result, :new_user_result) do
|
@@ -29,6 +41,9 @@ RSpec.shared_examples "a base model" do |ransack_permitted_attributes: [], ransa
|
|
29
41
|
[
|
30
42
|
:ransackable_associations, ransack_permitted_associations.map(&:to_s), lazy { new_user_ransack_permitted_associations.map(&:to_s) }
|
31
43
|
],
|
44
|
+
[
|
45
|
+
:ransackable_scopes, ransack_permitted_scopes.map(&:to_s), lazy { new_user_ransack_permitted_scopes.map(&:to_s) }
|
46
|
+
],
|
32
47
|
]
|
33
48
|
end
|
34
49
|
|
@@ -25,6 +25,7 @@ RSpec.shared_examples "a standard base model policy" do |factory, check_default_
|
|
25
25
|
[:search_fields],
|
26
26
|
[:permitted_associations_for_ransack],
|
27
27
|
[:permitted_attributes_for_ransack],
|
28
|
+
[:permitted_scopes_for_ransack],
|
28
29
|
[:editable_attributes],
|
29
30
|
[:permitted_attributes],
|
30
31
|
]
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: base_editing_bootstrap
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 1.
|
4
|
+
version: 1.8.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Marino Bonetti
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2025-
|
11
|
+
date: 2025-03-20 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rails
|
@@ -337,11 +337,15 @@ files:
|
|
337
337
|
- app/views/base_editing/_editing_form_help_text.html.erb
|
338
338
|
- app/views/base_editing/_editing_form_measure_unit.html.erb
|
339
339
|
- app/views/base_editing/_form.html.erb
|
340
|
+
- app/views/base_editing/_form_base_errors.html.erb
|
341
|
+
- app/views/base_editing/_form_body_container.html.erb
|
342
|
+
- app/views/base_editing/_form_container.html.erb
|
340
343
|
- app/views/base_editing/_form_field.html.erb
|
341
344
|
- app/views/base_editing/_form_field_container.html.erb
|
342
345
|
- app/views/base_editing/_form_field_header.html.erb
|
343
346
|
- app/views/base_editing/_form_fields_container.html.erb
|
344
347
|
- app/views/base_editing/_form_footer.html.erb
|
348
|
+
- app/views/base_editing/_form_footer_container.html.erb
|
345
349
|
- app/views/base_editing/_index_body.html.erb
|
346
350
|
- app/views/base_editing/_index_main_buttons.html.erb
|
347
351
|
- app/views/base_editing/_index_title_header.html.erb
|
@@ -368,6 +372,7 @@ files:
|
|
368
372
|
- app/views/base_editing/form_field/_enum.html.erb
|
369
373
|
- app/views/base_editing/form_field/_has_one_attachment.html.erb
|
370
374
|
- app/views/base_editing/form_field/_integer.html.erb
|
375
|
+
- app/views/base_editing/form_field/_textarea.html.erb
|
371
376
|
- app/views/base_editing/header_field/_base.html.erb
|
372
377
|
- app/views/base_editing/index.html.erb
|
373
378
|
- app/views/base_editing/new.html.erb
|