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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 0f292b335541e77a9d78ae97f8ef87cc436468094aaccd69750dae0f8dcf44af
4
- data.tar.gz: 7c3f97b3261d736258efc021f36cc131943763b13f7d933cfb0967cbf58e5f36
3
+ metadata.gz: 785488fc3726ebf47d41122c81fd74cfc1d25dcef6b5f3848d80689a5eb00e5b
4
+ data.tar.gz: e63fa0c1c288e38fde44df84260eefd133d726ae387e8742f549753a83394959
5
5
  SHA512:
6
- metadata.gz: 3ec5909dab19b68590b2e2beb5dbcf7b502f2bbeb9ae3c2c244117837c4eeb54d20848ca30381f0cae4c88f6047bf3d55c2dfeef37dd28623b7b59933970e33c
7
- data.tar.gz: dbaca2890a7a8a6ea1f9cde44fcd6d07495a6f289ba82f62698cc8b9cac6bfe20809133590ca05c91cecd1078ea7ce4a87087c0501e52570146eca17efd65486
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
- authorize base_class
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
- authorize @object
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
- authorize @object
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
- authorize @object
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 { "#{type}->#{generic_field}->#{template.short_identifier}->#{ locals.inspect}" }
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
- if valore
30
+ case valore
31
+ when true
30
32
  icon("check-lg", class: "text-success")
31
- else
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
- # Precedenza modello e campo specifico
22
- if lookup_context.exists?(field, [obj_base_path], true)
23
- return lookup_context.find(field, [obj_base_path], true)
24
- end
25
- # Ricerca tramite campo generico e prefissi di contesto che contiene anche controller e namespace di controller
26
- if lookup_context.exists?("#{base_path}/#{generic_field}", lookup_context.prefixes, true)
27
- view = lookup_context.find_all("#{base_path}/#{generic_field}", lookup_context.prefixes, true)
28
- return view.first
29
- end
30
- if lookup_context.exists?(generic_field, [obj_base_path], true)
31
- return lookup_context.find(generic_field, [obj_base_path], true)
32
- end
33
- if lookup_context.exists?("base_editing/#{base_path}/#{generic_field}", [], true)
34
- return lookup_context.find_all("base_editing/#{base_path}/#{generic_field}", [], true).first
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
 
@@ -24,6 +24,8 @@ class BaseModelPolicy < ApplicationPolicy
24
24
  []
25
25
  end
26
26
 
27
+ def permitted_scopes_for_ransack = []
28
+
27
29
  def search_fields = []
28
30
 
29
31
  def search_result_fields = []
@@ -6,7 +6,13 @@
6
6
  %>
7
7
  <%# locals: (object:,field:) -%>
8
8
  <%
9
- help_text = object.class.human_attribute_name("#{field}/help_text", default: "")
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
- <div class="card" id="card_id">
2
- <div class="card-header">
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
- <div class="card-body">
6
- <%= render partial: "form_field_header", local: {form:} %>
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
- </div>
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
- </div>
20
- </div>
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 %>
@@ -0,0 +1,3 @@
1
+ <div class="card-body">
2
+ <%= yield %>
3
+ </div>
@@ -0,0 +1,3 @@
1
+ <%= content_tag :div, class: "card", id: dom_id(@object, :form_container) do %>
2
+ <%= yield %>
3
+ <% end %>
@@ -2,18 +2,18 @@
2
2
  ##
3
3
  # Template per il rendering campo
4
4
  # - form -> FormBuilder
5
- # - form_field -> Array[String]
5
+ # - form_field -> String
6
6
  %>
7
7
  <%# locals: (form:, form_field:) -%>
8
8
  <%
9
- button_group_classes = ["mb-1"]
9
+ input_group_classes = ["mb-1"]
10
10
 
11
11
  content = form_print_field(form, form_field)
12
12
  unless content.match "checkbox"
13
- button_group_classes << "input-group"
13
+ input_group_classes << "input-group"
14
14
  end
15
- button_group_classes << (form.object.validated? ? "has-validation" : "")
16
- button_group_classes << "form-#{form_field}-input-group"
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: button_group_classes.join(" ") do %>
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) %>
@@ -0,0 +1,3 @@
1
+ <div class="card-footer">
2
+ <%= yield %>
3
+ </div>
@@ -0,0 +1,7 @@
1
+ <%
2
+ ##
3
+ # Template per il rendering del vero campo input
4
+ # eseguire override nei casi in cui si voglia renderizzare qualcosa di differente
5
+ %>
6
+ <%# locals: (form:, field:) -%>
7
+ <%= form.text_area(field) %>
@@ -76,3 +76,10 @@ it:
76
76
  ransack:
77
77
  predicates:
78
78
  i_cont: "contiene"
79
+ base_editing:
80
+ form_base_errors:
81
+ title: Presenti errori generici
82
+ post:
83
+ base_editing:
84
+ form_base_errors:
85
+ title: Presenti errori per il modello POST
@@ -1 +1 @@
1
- 1.6.0
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, :ransackable_associations, to: :@class
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 = ["form-control"]
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[:class] = "form-select #{html_options[:class]}"
50
- super(method, choices, options, html_options.merge(class: form_style_class_for(method, html_options)), &block)
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
- classes = (["form-check"] + (options.extract!(:class)[:class] || "").split(" ")).join(" ")
61
- @template.content_tag(:div, class: classes) do
62
- super(method, options.reverse_merge(class: "form-check-input"), checked_value, unchecked_value) + label_tag
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: true)
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/#{model_resource_name}.rb", class_name do
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: [] do
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).and(have_attributes(
93
- user: user,
94
- params: ActionController::Parameters.new(params).permit!
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( Pundit::VERSION) < Gem::Version.create('2.3.2')
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: [], ransack_permitted_associations: [], option_label_method: :to_s|
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.6.0
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-01-23 00:00:00.000000000 Z
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