Module: ActiveScaffold::Helpers::AssociationHelpers

Included in:
ViewHelpers
Defined in:
lib/active_scaffold/helpers/association_helpers.rb

Instance Method Summary collapse

Instance Method Details

#association_helper_method(association, method) ⇒ Object



15
16
17
18
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 15

def association_helper_method(association, method)
  model = association.inverse_klass
  override_helper_per_model(method, model)
end

#association_klass_scoped(association, klass, record) ⇒ Object



60
61
62
63
64
65
66
67
68
69
70
71
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 60

def association_klass_scoped(association, klass, record)
  if nested? && nested.through_association? && nested.child_association&.through_reflection == association
    # only ActiveRecord associations
    if nested.association.through_reflection.collection?
      nested_parent_record.send(nested.association.through_reflection.name)
    else
      klass.where(association.association_primary_key => nested_parent_record.send(nested.association.through_reflection.name)&.id)
    end
  else
    klass
  end
end

#association_options_count(association, conditions = nil) ⇒ Object



84
85
86
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 84

def association_options_count(association, conditions = nil)
  association.klass.where(conditions).count
end

#association_options_find(association, conditions = nil, klass = nil, record = nil) ⇒ Object

Provides a way to honor the :conditions on an association while searching the association’s klass



21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 21

def association_options_find(association, conditions = nil, klass = nil, record = nil)
  if klass.nil? && association.polymorphic?
    class_name = record.send(association.foreign_type) if association.belongs_to?
    return [] if class_name.blank?

    klass = class_name.constantize
    cache = !block_given?
  else
    cache = !block_given? && klass.nil?
    klass ||= association.klass
  end

  conditions ||= send(association_helper_method(association, :options_for_association_conditions), association, record)
  klass = send(association_helper_method(association, :association_klass_scoped), association, klass, record)
  cache_association_options(association, conditions, klass, cache) do
    relation = klass.where(conditions)
    column = column_for_association(association, record)
    if column&.includes
      include_assoc = column.includes.find { |assoc| assoc.is_a?(Hash) && assoc.include?(association.name) }
      relation = relation.includes(include_assoc[association.name]) if include_assoc
    end
    if column&.sort && column.sort&.dig(:sql)
      # with threasafe enabled, column.sort[:sql] returns proxied strings and
      # regexp capture won't work, which rails uses internally, so to_s is needed
      relation = relation.order(Array(column.sort[:sql]).map(&:to_s))
    end
    relation = yield(relation) if block_given?
    relation.to_a
  end
end

#cache_association_options(association, conditions, klass, cache = true) ⇒ Object

Cache the options for select



5
6
7
8
9
10
11
12
13
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 5

def cache_association_options(association, conditions, klass, cache = true)
  if active_scaffold_config.cache_association_options && cache
    @_associations_cache ||= Hash.new { |h, k| h[k] = {} }
    key = [association.name, association.inverse_klass.name, klass.respond_to?(:cache_key) ? klass.cache_key : klass.name].join('/')
    @_associations_cache[key][conditions] ||= yield
  else
    yield
  end
end

#column_for_association(association, record) ⇒ Object



52
53
54
55
56
57
58
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 52

def column_for_association(association, record)
  active_scaffold_config_for(record.class).columns[association.name]
rescue StandardError => e
  message = "Error on config for #{record.class.name}:"
  Rails.logger.warn "#{message}\n#{e.message}\n#{e.backtrace.join("\n")}"
  nil
end

#options_for_association_conditions(association, record = nil) ⇒ Object

A useful override for customizing the records present in an association dropdown. Should work in both the subform and form_ui=>:select modes. Check association.name to specialize the conditions per-column.



96
97
98
99
100
101
102
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 96

def options_for_association_conditions(association, record = nil)
  return nil if association.through?
  return nil unless association.has_one? || association.has_many?

  # Find only orphaned objects
  {association.foreign_key => nil}
end

#options_for_association_count(association, record) ⇒ Object



88
89
90
91
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 88

def options_for_association_count(association, record)
  conditions = send(association_helper_method(association, :options_for_association_conditions), association, record)
  association_options_count(association, conditions)
end

#record_select_params_for_add_existing(association, edit_associated_url_options, record) ⇒ Object



104
105
106
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 104

def record_select_params_for_add_existing(association, edit_associated_url_options, record)
  {onselect: "ActiveScaffold.record_select_onselect(#{url_for(edit_associated_url_options).to_json}, #{active_scaffold_id.to_json}, id);"}
end

#sorted_association_options_find(association, conditions = nil, record = nil) ⇒ Object

Sorts the options for select



74
75
76
77
78
79
80
81
82
# File 'lib/active_scaffold/helpers/association_helpers.rb', line 74

def sorted_association_options_find(association, conditions = nil, record = nil)
  options = association_options_find(association, conditions, nil, record)
  column = column_for_association(association, record)
  unless column&.sort && column.sort&.dig(:sql)
    method = column.options[:label_method] if column
    options = options.sort_by(&(method || :to_label).to_sym)
  end
  options
end